{
 "cells": [
  {
   "metadata": {},
   "cell_type": "markdown",
   "source": "# Downsampling",
   "id": "e37911f73aaefbfb"
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:19:13.482571Z",
     "start_time": "2025-09-16T19:19:13.478985Z"
    }
   },
   "cell_type": "code",
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n"
   ],
   "id": "78623523ab963865",
   "outputs": [],
   "execution_count": 4
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:19:14.653634Z",
     "start_time": "2025-09-16T19:19:14.147560Z"
    }
   },
   "cell_type": "code",
   "source": "data= pd.read_csv(\"emotions.csv\")",
   "id": "8d8ec0ce8b0b27f8",
   "outputs": [],
   "execution_count": 5
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:19:14.998627Z",
     "start_time": "2025-09-16T19:19:14.956240Z"
    }
   },
   "cell_type": "code",
   "source": [
    "df_sampled = (\n",
    "    data.groupby(\"label\", group_keys=False)\n",
    "      .apply(lambda x: x.sample(n=200, replace=False, random_state = 42))\n",
    "      .sample(frac=1, random_state=42)   # optional: shuffle all groups together\n",
    ").reset_index(drop = True)"
   ],
   "id": "650efd484c950911",
   "outputs": [],
   "execution_count": 6
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:19:23.035226Z",
     "start_time": "2025-09-16T19:19:23.029989Z"
    }
   },
   "cell_type": "code",
   "source": [
    "df_train = df_sampled.iloc[:1000, :]\n",
    "df_test = df_sampled.iloc[1000:, :]"
   ],
   "id": "adee3a4bc156de88",
   "outputs": [],
   "execution_count": 7
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:19:47.190679Z",
     "start_time": "2025-09-16T19:19:47.170334Z"
    }
   },
   "cell_type": "code",
   "source": [
    "df_train.to_csv(\"train_sentiment.csv\", index = False)\n",
    "df_test.to_csv(\"test_sentiment.csv\", index = False)"
   ],
   "id": "8bff5c769f0e6dcc",
   "outputs": [],
   "execution_count": 8
  },
  {
   "metadata": {},
   "cell_type": "markdown",
   "source": "# Generations",
   "id": "867102406bf87211"
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:21:51.792460Z",
     "start_time": "2025-09-16T19:21:51.146568Z"
    }
   },
   "cell_type": "code",
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "data_train = pd.read_csv(\"train_sentiment.csv\") # read data\n",
    "n = data_train.shape[0]\n",
    "print(n)"
   ],
   "id": "9ae03493086b01f2",
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1000\n"
     ]
    }
   ],
   "execution_count": 1
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:21:55.139751Z",
     "start_time": "2025-09-16T19:21:55.132413Z"
    }
   },
   "cell_type": "code",
   "source": [
    "label_map = {\n",
    "    0: \"sadness\",\n",
    "    1: \"joy\",\n",
    "    2: \"love\",\n",
    "    3: \"anger\",\n",
    "    4: \"fear\",\n",
    "    5: \"surprise\"\n",
    "}\n",
    "\n",
    "# Apply mapping\n",
    "data_train[\"emotion\"] = data_train[\"label\"].map(label_map)\n",
    "\n",
    "print(data_train.head)"
   ],
   "id": "b690128430453540",
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "<bound method NDFrame.head of                                                   text  label   emotion\n",
      "0    i only played the london mission from the demo...      5  surprise\n",
      "1    i am feeling fearful confused a deep and profo...      4      fear\n",
      "2    i go overboard with my sense of duty and respo...      0   sadness\n",
      "3    i feel most passionate about and currently thi...      2      love\n",
      "4    i feel almost embarrassed to be writing its be...      0   sadness\n",
      "..                                                 ...    ...       ...\n",
      "995  i started feeling super nauseous since im real...      1       joy\n",
      "996  i hate feeling envious and i see it as an alar...      3     anger\n",
      "997  im not quite so animated but its just a way of...      1       joy\n",
      "998  i am feeling very apprehensive i know it isnt ...      4      fear\n",
      "999     ive been feeling amorous which also kept me up      2      love\n",
      "\n",
      "[1000 rows x 3 columns]>\n"
     ]
    }
   ],
   "execution_count": 2
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:22:13.088736Z",
     "start_time": "2025-09-16T19:22:13.079275Z"
    }
   },
   "cell_type": "code",
   "source": [
    "emo_ls = \", \".join(data_train[\"emotion\"].drop_duplicates())\n",
    "emo_ls"
   ],
   "id": "3375256d84390f5d",
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'surprise, fear, sadness, love, joy, anger'"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "execution_count": 3
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:23:21.607694Z",
     "start_time": "2025-09-16T19:23:21.556568Z"
    }
   },
   "cell_type": "code",
   "source": [
    "import time\n",
    "import random\n",
    "import concurrent.futures\n",
    "import re\n",
    "from tenacity import retry, stop_after_attempt, wait_exponential, retry_if_exception_type\n",
    "import openai\n",
    "from openai import OpenAI\n",
    "\n",
    "client = OpenAI(api_key=\"...\")\n",
    "\n",
    "class RateLimitException(Exception):\n",
    "    pass\n",
    "\n",
    "@retry(\n",
    "    retry=retry_if_exception_type(RateLimitException),\n",
    "    wait=wait_exponential(multiplier=1, min=1, max=60),\n",
    "    stop=stop_after_attempt(15)\n",
    ")\n",
    "def call_openai_api(client, prompt, n=20):\n",
    "    try:\n",
    "        response = client.chat.completions.create(\n",
    "            model=\"gpt-4.1-nano\",\n",
    "            messages=[{\"role\": \"user\", \"content\": prompt}],\n",
    "            n=n,\n",
    "            temperature=1.5\n",
    "        )\n",
    "        return response\n",
    "    except Exception as e:\n",
    "        error_message = str(e).lower()\n",
    "        if \"rate limit\" in error_message or \"429\" in error_message:\n",
    "            wait_time_match = re.search(r'try again in (\\d+)ms', error_message)\n",
    "            if wait_time_match:\n",
    "                wait_ms = int(wait_time_match.group(1))\n",
    "                wait_time = (wait_ms / 1000) + random.uniform(0.1, 0.5)\n",
    "            else:\n",
    "                wait_time = random.uniform(1, 3)\n",
    "\n",
    "            print(f\"Rate limit hit. Waiting for {wait_time:.2f} seconds before retry...\")\n",
    "            time.sleep(wait_time)\n",
    "            raise RateLimitException(\"Rate limit exceeded\")\n",
    "        else:\n",
    "            raise\n",
    "\n",
    "import re\n",
    "\n",
    "def count_sentences(text: str) -> list[str]:\n",
    "    \"\"\"\n",
    "    Split text into sentences conservatively.\n",
    "    Returns a list of non-empty sentences.\n",
    "    \"\"\"\n",
    "    # Split on ., ?, ! followed by space/newline or end of string\n",
    "    parts = re.split(r'(?<=[.!?])\\s+(?=[A-Z0-9(])|(?<=[.!?])$', text.strip())\n",
    "    # Clean up stray empties/whitespace\n",
    "    sents = [s.strip() for s in parts if s and s.strip()]\n",
    "    return sents\n",
    "\n",
    "def process_item(client, idx, sentences):\n",
    "    tweet = sentences[0]\n",
    "    base_prompt = (\n",
    "        \"You are given a short Twitter message.\\n\\n\"\n",
    "        f\"Message: {tweet}\\n\\n\"\n",
    "        \"Task: Extend the message by writing additional content that keeps the SAME sentiment and topic consistent.\\n\"\n",
    "        \"- Write EXACTLY 5 sentences.\\n\"\n",
    "        \"- Paraphrase and expand naturally; do not reuse wording from the original.\\n\"\n",
    "        \"- Introduce diversity in phrasing, tone, and detail while staying aligned with the sentiment.\\n\"\n",
    "        \"- Avoid lists, bullets, hashtags, mentions, links, or numbering; just 5 full sentences in a single paragraph.\\n\"\n",
    "        \"- No disclaimers, no citations, no markdown.\\n\"\n",
    "    )\n",
    "\n",
    "    max_attempts = 20\n",
    "    while max_attempts > 0:\n",
    "        try:\n",
    "            response = call_openai_api(client, base_prompt, n=1)\n",
    "            content = response.choices[0].message.content.strip()\n",
    "            sents = count_sentences(content)\n",
    "\n",
    "            if len(sents) == 5:\n",
    "                # Return clean paragraph\n",
    "                return sents\n",
    "            else:\n",
    "                # Not exactly 10 → retry\n",
    "                max_attempts -= 1\n",
    "        except Exception:\n",
    "            max_attempts -= 1\n",
    "\n",
    "    print(f\"Warning: Could not get exactly 5 sentences for item {idx}\")\n",
    "    return None\n",
    "\n",
    "def process_with_rate_limiting(client, input_texts, max_concurrent=5, batch_size=20):\n",
    "    all_responses = []\n",
    "\n",
    "    for batch_start in range(0, len(input_texts), batch_size):\n",
    "        batch_end = min(batch_start + batch_size, len(input_texts))\n",
    "        batch = input_texts[batch_start:batch_end]\n",
    "\n",
    "        print(f\"Processing batch {batch_start // batch_size + 1}, items {batch_start} to {batch_end - 1}\")\n",
    "\n",
    "        with concurrent.futures.ThreadPoolExecutor(max_workers=max_concurrent) as executor:\n",
    "            futures = [\n",
    "                executor.submit(process_item, client, idx + batch_start, sentences)\n",
    "                for idx, sentences in enumerate(batch)\n",
    "            ]\n",
    "\n",
    "            # Ensure order of responses matches order of input_texts\n",
    "            batch_results = [future.result() for future in futures]\n",
    "\n",
    "        all_responses.extend(batch_results)\n",
    "\n",
    "        if batch_end < len(input_texts):\n",
    "            wait_time = random.uniform(1, 3)\n",
    "            time.sleep(wait_time)\n",
    "\n",
    "    return all_responses"
   ],
   "id": "4701be79926dca3",
   "outputs": [],
   "execution_count": 5
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:23:56.895786Z",
     "start_time": "2025-09-16T19:23:56.876262Z"
    }
   },
   "cell_type": "code",
   "source": [
    "pairs = []\n",
    "for i in range(data_train.shape[0]):\n",
    "    pairs.append((data_train[\"text\"].iloc[i], data_train[\"emotion\"].iloc[i]))"
   ],
   "id": "653410743b15f11e",
   "outputs": [],
   "execution_count": 7
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:35:25.672058Z",
     "start_time": "2025-09-16T19:24:19.867292Z"
    }
   },
   "cell_type": "code",
   "source": "generations = process_with_rate_limiting(client, pairs)\n",
   "id": "38d1b1efc3e8a5d1",
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Processing batch 1, items 0 to 19\n",
      "Processing batch 2, items 20 to 39\n",
      "Processing batch 3, items 40 to 59\n",
      "Processing batch 4, items 60 to 79\n",
      "Processing batch 5, items 80 to 99\n",
      "Processing batch 6, items 100 to 119\n",
      "Processing batch 7, items 120 to 139\n",
      "Processing batch 8, items 140 to 159\n",
      "Processing batch 9, items 160 to 179\n",
      "Processing batch 10, items 180 to 199\n",
      "Processing batch 11, items 200 to 219\n",
      "Processing batch 12, items 220 to 239\n",
      "Processing batch 13, items 240 to 259\n",
      "Processing batch 14, items 260 to 279\n",
      "Processing batch 15, items 280 to 299\n",
      "Processing batch 16, items 300 to 319\n",
      "Processing batch 17, items 320 to 339\n",
      "Processing batch 18, items 340 to 359\n",
      "Processing batch 19, items 360 to 379\n",
      "Processing batch 20, items 380 to 399\n",
      "Processing batch 21, items 400 to 419\n",
      "Processing batch 22, items 420 to 439\n",
      "Processing batch 23, items 440 to 459\n",
      "Processing batch 24, items 460 to 479\n",
      "Processing batch 25, items 480 to 499\n",
      "Processing batch 26, items 500 to 519\n",
      "Processing batch 27, items 520 to 539\n",
      "Processing batch 28, items 540 to 559\n",
      "Processing batch 29, items 560 to 579\n",
      "Processing batch 30, items 580 to 599\n",
      "Processing batch 31, items 600 to 619\n",
      "Processing batch 32, items 620 to 639\n",
      "Processing batch 33, items 640 to 659\n",
      "Processing batch 34, items 660 to 679\n",
      "Processing batch 35, items 680 to 699\n",
      "Processing batch 36, items 700 to 719\n",
      "Processing batch 37, items 720 to 739\n",
      "Processing batch 38, items 740 to 759\n",
      "Processing batch 39, items 760 to 779\n",
      "Processing batch 40, items 780 to 799\n",
      "Processing batch 41, items 800 to 819\n",
      "Processing batch 42, items 820 to 839\n",
      "Processing batch 43, items 840 to 859\n",
      "Processing batch 44, items 860 to 879\n",
      "Processing batch 45, items 880 to 899\n",
      "Processing batch 46, items 900 to 919\n",
      "Processing batch 47, items 920 to 939\n",
      "Processing batch 48, items 940 to 959\n",
      "Processing batch 49, items 960 to 979\n",
      "Processing batch 50, items 980 to 999\n"
     ]
    }
   ],
   "execution_count": 9
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:36:43.785714Z",
     "start_time": "2025-09-16T19:36:43.772812Z"
    }
   },
   "cell_type": "code",
   "source": [
    "import pickle\n",
    "\n",
    "with open(\"generations_sentiment.pickle\", \"wb\") as file:\n",
    "    pickle.dump(generations, file)"
   ],
   "id": "29045ed3ee12203c",
   "outputs": [],
   "execution_count": 10
  },
  {
   "metadata": {},
   "cell_type": "markdown",
   "source": "# Gemini as Judge",
   "id": "b4e6e0878942f0b8"
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:37:54.645989Z",
     "start_time": "2025-09-16T19:37:54.635008Z"
    }
   },
   "cell_type": "code",
   "source": [
    "import google.generativeai as genai\n",
    "import os\n",
    "import time\n",
    "from dotenv import load_dotenv\n",
    "\n",
    "# Load environment variables\n",
    "load_dotenv()\n",
    "genai.configure(api_key=os.getenv(\"GOOGLE_API_KEY\"))\n",
    "\n",
    "model = genai.GenerativeModel(\"gemini-2.5-flash\")\n",
    "\n",
    "def batch_eval(all_pairs, disease_ls, batch_size=20, max_retries=5, sleep_time=2, save_every=25, save_path=\"scores_checkpoint.csv\"):\n",
    "    all_scores = []\n",
    "    batch_counter = 0  # track how many batches processed\n",
    "\n",
    "    for start in range(0, all_pairs.shape[0], batch_size):\n",
    "        end = min(start + batch_size, all_pairs.shape[0])\n",
    "        batch = all_pairs.iloc[start:end]\n",
    "\n",
    "        # Build base prompt\n",
    "# Build base prompt\n",
    "        prompt = (\n",
    "            \"You are evaluating individual sentences for sentiment consistency.\\n\\n\"\n",
    "            \"Scoring instructions:\\n\"\n",
    "            \"- Assign each sentence a score between 0 and 1, rounded to two decimal places.\\n\"\n",
    "            \"- Criteria: The sentence should clearly express the SPECIFIED sentiment, remain coherent, and avoid conflicting or opposite emotions.\\n\"\n",
    "            \"- Use the full 0–1 range: 1 = perfectly consistent with the sentiment and natural; 0 = completely unusable.\\n\"\n",
    "            \"- 0.5 is the threshold: any sentence with a score ≤ 0.5 should be dropped to prevent sentiment drift.\\n\\n\"\n",
    "            \"Output requirements:\\n\"\n",
    "            \"- Output only the scores, one per line, in the same order as the input cases.\\n\"\n",
    "            \"- Do not include explanations, text, or formatting other than the numeric scores.\\n\\n\"\n",
    "        )\n",
    "\n",
    "        for i, row in batch.iterrows():\n",
    "            sent, senti = row[\"sentence\"], row[\"sentiment\"]\n",
    "            prompt += f\"Case {i}:\\nSentiment: {senti}\\nSentence: {sent}\\n\"\n",
    "\n",
    "        # Retry loop\n",
    "        scores = []\n",
    "        for attempt in range(1, max_retries + 1):\n",
    "            try:\n",
    "                response = model.generate_content(prompt)\n",
    "                scores = response.text.strip().splitlines()\n",
    "\n",
    "                if len(scores) == len(batch):\n",
    "                    break  # ✅ got the right number of outputs\n",
    "                else:\n",
    "                    print(\n",
    "                        f\"⚠️ Attempt {attempt}: Expected {len(batch)} scores, got {len(scores)}. Retrying...\"\n",
    "                    )\n",
    "                    time.sleep(sleep_time)\n",
    "\n",
    "            except Exception as e:\n",
    "                print(f\"❌ Error on attempt {attempt}: {e}\")\n",
    "                time.sleep(sleep_time)\n",
    "\n",
    "        if len(scores) != len(batch):\n",
    "            raise ValueError(\n",
    "                f\"Failed after {max_retries} retries: Expected {len(batch)} scores, got {len(scores)}\"\n",
    "            )\n",
    "\n",
    "        all_scores.extend(scores)\n",
    "        batch_counter += 1\n",
    "        print(f\"✅ Processed {end}/{all_pairs.shape[0]}\")\n",
    "\n",
    "        if batch_counter % save_every == 0:\n",
    "            pd.DataFrame({\"score\": all_scores}).to_csv(save_path, index=False)\n",
    "            print(f\"💾 Saved checkpoint after {batch_counter} batches at {save_path}\")\n",
    "\n",
    "    pd.DataFrame({\"score\": all_scores}).to_csv(save_path, index=False)\n",
    "    print(f\"🎉 Finished. Final results saved at {save_path}\")\n",
    "\n",
    "    return all_scores"
   ],
   "id": "744693875fcaeade",
   "outputs": [],
   "execution_count": 17
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T19:37:57.265893Z",
     "start_time": "2025-09-16T19:37:57.245806Z"
    }
   },
   "cell_type": "code",
   "source": [
    "\n",
    "import pickle\n",
    "with open(\"generations_sentiment.pickle\", \"rb\") as file:\n",
    "    generations = pickle.load(file)\n",
    "\n",
    "all_rows = []\n",
    "\n",
    "# 2. Loop through the augmented data.\n",
    "for i in range(data_train.shape[0]):\n",
    "    output_text = data_train[\"emotion\"].iloc[i]\n",
    "\n",
    "    # Add the original row\n",
    "    for j in generations[i]:\n",
    "        all_rows.append([j, output_text])\n",
    "\n",
    "\n",
    "\n",
    "# 4. Create the DataFrame from the list in one single, efficient operation.\n",
    "df = pd.DataFrame(all_rows, columns=[\"sentence\", \"sentiment\"])"
   ],
   "id": "92a09c870c03d26d",
   "outputs": [],
   "execution_count": 18
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T21:25:55.862531Z",
     "start_time": "2025-09-16T19:42:49.367821Z"
    }
   },
   "cell_type": "code",
   "source": "scores = batch_eval(df, emo_ls, batch_size=10, save_path=\"scores_flash_sentiment.csv\", save_every = 10)",
   "id": "7694a01496fc49a1",
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✅ Processed 10/4900\n",
      "✅ Processed 20/4900\n",
      "✅ Processed 30/4900\n",
      "✅ Processed 40/4900\n",
      "✅ Processed 50/4900\n",
      "✅ Processed 60/4900\n",
      "✅ Processed 70/4900\n",
      "✅ Processed 80/4900\n",
      "✅ Processed 90/4900\n",
      "✅ Processed 100/4900\n",
      "💾 Saved checkpoint after 10 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 110/4900\n",
      "✅ Processed 120/4900\n",
      "✅ Processed 130/4900\n",
      "✅ Processed 140/4900\n",
      "✅ Processed 150/4900\n",
      "✅ Processed 160/4900\n",
      "✅ Processed 170/4900\n",
      "✅ Processed 180/4900\n",
      "✅ Processed 190/4900\n",
      "✅ Processed 200/4900\n",
      "💾 Saved checkpoint after 20 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 210/4900\n",
      "✅ Processed 220/4900\n",
      "✅ Processed 230/4900\n",
      "✅ Processed 240/4900\n",
      "✅ Processed 250/4900\n",
      "✅ Processed 260/4900\n",
      "✅ Processed 270/4900\n",
      "✅ Processed 280/4900\n",
      "✅ Processed 290/4900\n",
      "✅ Processed 300/4900\n",
      "💾 Saved checkpoint after 30 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 310/4900\n",
      "✅ Processed 320/4900\n",
      "✅ Processed 330/4900\n",
      "✅ Processed 340/4900\n",
      "✅ Processed 350/4900\n",
      "✅ Processed 360/4900\n",
      "✅ Processed 370/4900\n",
      "✅ Processed 380/4900\n",
      "✅ Processed 390/4900\n",
      "✅ Processed 400/4900\n",
      "💾 Saved checkpoint after 40 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 410/4900\n",
      "✅ Processed 420/4900\n",
      "✅ Processed 430/4900\n",
      "✅ Processed 440/4900\n",
      "✅ Processed 450/4900\n",
      "✅ Processed 460/4900\n",
      "✅ Processed 470/4900\n",
      "✅ Processed 480/4900\n",
      "✅ Processed 490/4900\n",
      "✅ Processed 500/4900\n",
      "💾 Saved checkpoint after 50 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 510/4900\n",
      "✅ Processed 520/4900\n",
      "✅ Processed 530/4900\n",
      "✅ Processed 540/4900\n",
      "✅ Processed 550/4900\n",
      "✅ Processed 560/4900\n",
      "✅ Processed 570/4900\n",
      "✅ Processed 580/4900\n",
      "✅ Processed 590/4900\n",
      "✅ Processed 600/4900\n",
      "💾 Saved checkpoint after 60 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 610/4900\n",
      "✅ Processed 620/4900\n",
      "✅ Processed 630/4900\n",
      "✅ Processed 640/4900\n",
      "✅ Processed 650/4900\n",
      "✅ Processed 660/4900\n",
      "✅ Processed 670/4900\n",
      "✅ Processed 680/4900\n",
      "✅ Processed 690/4900\n",
      "✅ Processed 700/4900\n",
      "💾 Saved checkpoint after 70 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 710/4900\n",
      "✅ Processed 720/4900\n",
      "✅ Processed 730/4900\n",
      "✅ Processed 740/4900\n",
      "✅ Processed 750/4900\n",
      "✅ Processed 760/4900\n",
      "✅ Processed 770/4900\n",
      "✅ Processed 780/4900\n",
      "✅ Processed 790/4900\n",
      "✅ Processed 800/4900\n",
      "💾 Saved checkpoint after 80 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 810/4900\n",
      "✅ Processed 820/4900\n",
      "✅ Processed 830/4900\n",
      "✅ Processed 840/4900\n",
      "✅ Processed 850/4900\n",
      "✅ Processed 860/4900\n",
      "✅ Processed 870/4900\n",
      "✅ Processed 880/4900\n",
      "✅ Processed 890/4900\n",
      "✅ Processed 900/4900\n",
      "💾 Saved checkpoint after 90 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 910/4900\n",
      "✅ Processed 920/4900\n",
      "✅ Processed 930/4900\n",
      "✅ Processed 940/4900\n",
      "✅ Processed 950/4900\n",
      "✅ Processed 960/4900\n",
      "✅ Processed 970/4900\n",
      "✅ Processed 980/4900\n",
      "✅ Processed 990/4900\n",
      "✅ Processed 1000/4900\n",
      "💾 Saved checkpoint after 100 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1010/4900\n",
      "✅ Processed 1020/4900\n",
      "✅ Processed 1030/4900\n",
      "✅ Processed 1040/4900\n",
      "✅ Processed 1050/4900\n",
      "✅ Processed 1060/4900\n",
      "✅ Processed 1070/4900\n",
      "✅ Processed 1080/4900\n",
      "✅ Processed 1090/4900\n",
      "✅ Processed 1100/4900\n",
      "💾 Saved checkpoint after 110 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1110/4900\n",
      "✅ Processed 1120/4900\n",
      "✅ Processed 1130/4900\n",
      "✅ Processed 1140/4900\n",
      "✅ Processed 1150/4900\n",
      "✅ Processed 1160/4900\n",
      "✅ Processed 1170/4900\n",
      "✅ Processed 1180/4900\n",
      "✅ Processed 1190/4900\n",
      "✅ Processed 1200/4900\n",
      "💾 Saved checkpoint after 120 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1210/4900\n",
      "✅ Processed 1220/4900\n",
      "✅ Processed 1230/4900\n",
      "✅ Processed 1240/4900\n",
      "✅ Processed 1250/4900\n",
      "✅ Processed 1260/4900\n",
      "✅ Processed 1270/4900\n",
      "✅ Processed 1280/4900\n",
      "✅ Processed 1290/4900\n",
      "✅ Processed 1300/4900\n",
      "💾 Saved checkpoint after 130 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1310/4900\n",
      "✅ Processed 1320/4900\n",
      "✅ Processed 1330/4900\n",
      "✅ Processed 1340/4900\n",
      "✅ Processed 1350/4900\n",
      "✅ Processed 1360/4900\n",
      "✅ Processed 1370/4900\n",
      "✅ Processed 1380/4900\n",
      "✅ Processed 1390/4900\n",
      "✅ Processed 1400/4900\n",
      "💾 Saved checkpoint after 140 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1410/4900\n",
      "✅ Processed 1420/4900\n",
      "✅ Processed 1430/4900\n",
      "✅ Processed 1440/4900\n",
      "✅ Processed 1450/4900\n",
      "✅ Processed 1460/4900\n",
      "✅ Processed 1470/4900\n",
      "✅ Processed 1480/4900\n",
      "✅ Processed 1490/4900\n",
      "✅ Processed 1500/4900\n",
      "💾 Saved checkpoint after 150 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1510/4900\n",
      "✅ Processed 1520/4900\n",
      "✅ Processed 1530/4900\n",
      "✅ Processed 1540/4900\n",
      "✅ Processed 1550/4900\n",
      "✅ Processed 1560/4900\n",
      "✅ Processed 1570/4900\n",
      "✅ Processed 1580/4900\n",
      "✅ Processed 1590/4900\n",
      "✅ Processed 1600/4900\n",
      "💾 Saved checkpoint after 160 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1610/4900\n",
      "✅ Processed 1620/4900\n",
      "✅ Processed 1630/4900\n",
      "✅ Processed 1640/4900\n",
      "✅ Processed 1650/4900\n",
      "✅ Processed 1660/4900\n",
      "✅ Processed 1670/4900\n",
      "✅ Processed 1680/4900\n",
      "✅ Processed 1690/4900\n",
      "✅ Processed 1700/4900\n",
      "💾 Saved checkpoint after 170 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1710/4900\n",
      "✅ Processed 1720/4900\n",
      "✅ Processed 1730/4900\n",
      "✅ Processed 1740/4900\n",
      "✅ Processed 1750/4900\n",
      "✅ Processed 1760/4900\n",
      "✅ Processed 1770/4900\n",
      "✅ Processed 1780/4900\n",
      "✅ Processed 1790/4900\n",
      "✅ Processed 1800/4900\n",
      "💾 Saved checkpoint after 180 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1810/4900\n",
      "✅ Processed 1820/4900\n",
      "✅ Processed 1830/4900\n",
      "✅ Processed 1840/4900\n",
      "✅ Processed 1850/4900\n",
      "✅ Processed 1860/4900\n",
      "✅ Processed 1870/4900\n",
      "✅ Processed 1880/4900\n",
      "✅ Processed 1890/4900\n",
      "✅ Processed 1900/4900\n",
      "💾 Saved checkpoint after 190 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 1910/4900\n",
      "✅ Processed 1920/4900\n",
      "✅ Processed 1930/4900\n",
      "✅ Processed 1940/4900\n",
      "✅ Processed 1950/4900\n",
      "✅ Processed 1960/4900\n",
      "✅ Processed 1970/4900\n",
      "✅ Processed 1980/4900\n",
      "✅ Processed 1990/4900\n",
      "✅ Processed 2000/4900\n",
      "💾 Saved checkpoint after 200 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2010/4900\n",
      "✅ Processed 2020/4900\n",
      "✅ Processed 2030/4900\n",
      "✅ Processed 2040/4900\n",
      "✅ Processed 2050/4900\n",
      "✅ Processed 2060/4900\n",
      "✅ Processed 2070/4900\n",
      "✅ Processed 2080/4900\n",
      "✅ Processed 2090/4900\n",
      "✅ Processed 2100/4900\n",
      "💾 Saved checkpoint after 210 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2110/4900\n",
      "✅ Processed 2120/4900\n",
      "✅ Processed 2130/4900\n",
      "✅ Processed 2140/4900\n",
      "✅ Processed 2150/4900\n",
      "✅ Processed 2160/4900\n",
      "✅ Processed 2170/4900\n",
      "✅ Processed 2180/4900\n",
      "✅ Processed 2190/4900\n",
      "✅ Processed 2200/4900\n",
      "💾 Saved checkpoint after 220 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2210/4900\n",
      "✅ Processed 2220/4900\n",
      "✅ Processed 2230/4900\n",
      "✅ Processed 2240/4900\n",
      "✅ Processed 2250/4900\n",
      "✅ Processed 2260/4900\n",
      "✅ Processed 2270/4900\n",
      "✅ Processed 2280/4900\n",
      "✅ Processed 2290/4900\n",
      "✅ Processed 2300/4900\n",
      "💾 Saved checkpoint after 230 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2310/4900\n",
      "✅ Processed 2320/4900\n",
      "✅ Processed 2330/4900\n",
      "✅ Processed 2340/4900\n",
      "✅ Processed 2350/4900\n",
      "✅ Processed 2360/4900\n",
      "✅ Processed 2370/4900\n",
      "✅ Processed 2380/4900\n",
      "✅ Processed 2390/4900\n",
      "✅ Processed 2400/4900\n",
      "💾 Saved checkpoint after 240 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2410/4900\n",
      "✅ Processed 2420/4900\n",
      "✅ Processed 2430/4900\n",
      "✅ Processed 2440/4900\n",
      "✅ Processed 2450/4900\n",
      "✅ Processed 2460/4900\n",
      "✅ Processed 2470/4900\n",
      "✅ Processed 2480/4900\n",
      "✅ Processed 2490/4900\n",
      "✅ Processed 2500/4900\n",
      "💾 Saved checkpoint after 250 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2510/4900\n",
      "✅ Processed 2520/4900\n",
      "✅ Processed 2530/4900\n",
      "✅ Processed 2540/4900\n",
      "✅ Processed 2550/4900\n",
      "✅ Processed 2560/4900\n",
      "✅ Processed 2570/4900\n",
      "✅ Processed 2580/4900\n",
      "✅ Processed 2590/4900\n",
      "✅ Processed 2600/4900\n",
      "💾 Saved checkpoint after 260 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2610/4900\n",
      "✅ Processed 2620/4900\n",
      "✅ Processed 2630/4900\n",
      "✅ Processed 2640/4900\n",
      "✅ Processed 2650/4900\n",
      "✅ Processed 2660/4900\n",
      "✅ Processed 2670/4900\n",
      "✅ Processed 2680/4900\n",
      "✅ Processed 2690/4900\n",
      "✅ Processed 2700/4900\n",
      "💾 Saved checkpoint after 270 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2710/4900\n",
      "✅ Processed 2720/4900\n",
      "✅ Processed 2730/4900\n",
      "✅ Processed 2740/4900\n",
      "✅ Processed 2750/4900\n",
      "✅ Processed 2760/4900\n",
      "✅ Processed 2770/4900\n",
      "✅ Processed 2780/4900\n",
      "✅ Processed 2790/4900\n",
      "✅ Processed 2800/4900\n",
      "💾 Saved checkpoint after 280 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2810/4900\n",
      "✅ Processed 2820/4900\n",
      "✅ Processed 2830/4900\n",
      "✅ Processed 2840/4900\n",
      "✅ Processed 2850/4900\n",
      "✅ Processed 2860/4900\n",
      "✅ Processed 2870/4900\n",
      "✅ Processed 2880/4900\n",
      "✅ Processed 2890/4900\n",
      "✅ Processed 2900/4900\n",
      "💾 Saved checkpoint after 290 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 2910/4900\n",
      "✅ Processed 2920/4900\n",
      "✅ Processed 2930/4900\n",
      "✅ Processed 2940/4900\n",
      "✅ Processed 2950/4900\n",
      "✅ Processed 2960/4900\n",
      "✅ Processed 2970/4900\n",
      "✅ Processed 2980/4900\n",
      "✅ Processed 2990/4900\n",
      "✅ Processed 3000/4900\n",
      "💾 Saved checkpoint after 300 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3010/4900\n",
      "✅ Processed 3020/4900\n",
      "✅ Processed 3030/4900\n",
      "✅ Processed 3040/4900\n",
      "✅ Processed 3050/4900\n",
      "✅ Processed 3060/4900\n",
      "✅ Processed 3070/4900\n",
      "✅ Processed 3080/4900\n",
      "✅ Processed 3090/4900\n",
      "✅ Processed 3100/4900\n",
      "💾 Saved checkpoint after 310 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3110/4900\n",
      "✅ Processed 3120/4900\n",
      "✅ Processed 3130/4900\n",
      "✅ Processed 3140/4900\n",
      "✅ Processed 3150/4900\n",
      "✅ Processed 3160/4900\n",
      "✅ Processed 3170/4900\n",
      "✅ Processed 3180/4900\n",
      "✅ Processed 3190/4900\n",
      "✅ Processed 3200/4900\n",
      "💾 Saved checkpoint after 320 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3210/4900\n",
      "✅ Processed 3220/4900\n",
      "✅ Processed 3230/4900\n",
      "✅ Processed 3240/4900\n",
      "✅ Processed 3250/4900\n",
      "✅ Processed 3260/4900\n",
      "✅ Processed 3270/4900\n",
      "✅ Processed 3280/4900\n",
      "✅ Processed 3290/4900\n",
      "✅ Processed 3300/4900\n",
      "💾 Saved checkpoint after 330 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3310/4900\n",
      "✅ Processed 3320/4900\n",
      "✅ Processed 3330/4900\n",
      "✅ Processed 3340/4900\n",
      "✅ Processed 3350/4900\n",
      "✅ Processed 3360/4900\n",
      "✅ Processed 3370/4900\n",
      "✅ Processed 3380/4900\n",
      "✅ Processed 3390/4900\n",
      "✅ Processed 3400/4900\n",
      "💾 Saved checkpoint after 340 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3410/4900\n",
      "✅ Processed 3420/4900\n",
      "✅ Processed 3430/4900\n",
      "✅ Processed 3440/4900\n",
      "✅ Processed 3450/4900\n",
      "✅ Processed 3460/4900\n",
      "✅ Processed 3470/4900\n",
      "✅ Processed 3480/4900\n",
      "✅ Processed 3490/4900\n",
      "✅ Processed 3500/4900\n",
      "💾 Saved checkpoint after 350 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3510/4900\n",
      "✅ Processed 3520/4900\n",
      "✅ Processed 3530/4900\n",
      "✅ Processed 3540/4900\n",
      "✅ Processed 3550/4900\n",
      "✅ Processed 3560/4900\n",
      "✅ Processed 3570/4900\n",
      "✅ Processed 3580/4900\n",
      "✅ Processed 3590/4900\n",
      "✅ Processed 3600/4900\n",
      "💾 Saved checkpoint after 360 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3610/4900\n",
      "✅ Processed 3620/4900\n",
      "✅ Processed 3630/4900\n",
      "✅ Processed 3640/4900\n",
      "✅ Processed 3650/4900\n",
      "✅ Processed 3660/4900\n",
      "✅ Processed 3670/4900\n",
      "✅ Processed 3680/4900\n",
      "✅ Processed 3690/4900\n",
      "✅ Processed 3700/4900\n",
      "💾 Saved checkpoint after 370 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3710/4900\n",
      "✅ Processed 3720/4900\n",
      "✅ Processed 3730/4900\n",
      "✅ Processed 3740/4900\n",
      "✅ Processed 3750/4900\n",
      "✅ Processed 3760/4900\n",
      "✅ Processed 3770/4900\n",
      "✅ Processed 3780/4900\n",
      "✅ Processed 3790/4900\n",
      "✅ Processed 3800/4900\n",
      "💾 Saved checkpoint after 380 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3810/4900\n",
      "✅ Processed 3820/4900\n",
      "✅ Processed 3830/4900\n",
      "✅ Processed 3840/4900\n",
      "✅ Processed 3850/4900\n",
      "✅ Processed 3860/4900\n",
      "✅ Processed 3870/4900\n",
      "✅ Processed 3880/4900\n",
      "✅ Processed 3890/4900\n",
      "✅ Processed 3900/4900\n",
      "💾 Saved checkpoint after 390 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 3910/4900\n",
      "✅ Processed 3920/4900\n",
      "✅ Processed 3930/4900\n",
      "✅ Processed 3940/4900\n",
      "✅ Processed 3950/4900\n",
      "✅ Processed 3960/4900\n",
      "✅ Processed 3970/4900\n",
      "✅ Processed 3980/4900\n",
      "✅ Processed 3990/4900\n",
      "✅ Processed 4000/4900\n",
      "💾 Saved checkpoint after 400 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 4010/4900\n",
      "✅ Processed 4020/4900\n",
      "✅ Processed 4030/4900\n",
      "✅ Processed 4040/4900\n",
      "✅ Processed 4050/4900\n",
      "✅ Processed 4060/4900\n",
      "✅ Processed 4070/4900\n",
      "✅ Processed 4080/4900\n",
      "✅ Processed 4090/4900\n",
      "✅ Processed 4100/4900\n",
      "💾 Saved checkpoint after 410 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 4110/4900\n",
      "✅ Processed 4120/4900\n",
      "✅ Processed 4130/4900\n",
      "✅ Processed 4140/4900\n",
      "✅ Processed 4150/4900\n",
      "✅ Processed 4160/4900\n",
      "✅ Processed 4170/4900\n",
      "✅ Processed 4180/4900\n",
      "✅ Processed 4190/4900\n",
      "✅ Processed 4200/4900\n",
      "💾 Saved checkpoint after 420 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 4210/4900\n",
      "✅ Processed 4220/4900\n",
      "✅ Processed 4230/4900\n",
      "✅ Processed 4240/4900\n",
      "✅ Processed 4250/4900\n",
      "✅ Processed 4260/4900\n",
      "✅ Processed 4270/4900\n",
      "✅ Processed 4280/4900\n",
      "✅ Processed 4290/4900\n",
      "✅ Processed 4300/4900\n",
      "💾 Saved checkpoint after 430 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 4310/4900\n",
      "✅ Processed 4320/4900\n",
      "✅ Processed 4330/4900\n",
      "✅ Processed 4340/4900\n",
      "✅ Processed 4350/4900\n",
      "✅ Processed 4360/4900\n",
      "✅ Processed 4370/4900\n",
      "✅ Processed 4380/4900\n",
      "✅ Processed 4390/4900\n",
      "✅ Processed 4400/4900\n",
      "💾 Saved checkpoint after 440 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 4410/4900\n",
      "✅ Processed 4420/4900\n",
      "✅ Processed 4430/4900\n",
      "✅ Processed 4440/4900\n",
      "✅ Processed 4450/4900\n",
      "✅ Processed 4460/4900\n",
      "✅ Processed 4470/4900\n",
      "✅ Processed 4480/4900\n",
      "✅ Processed 4490/4900\n",
      "✅ Processed 4500/4900\n",
      "💾 Saved checkpoint after 450 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 4510/4900\n",
      "✅ Processed 4520/4900\n",
      "✅ Processed 4530/4900\n",
      "✅ Processed 4540/4900\n",
      "✅ Processed 4550/4900\n",
      "✅ Processed 4560/4900\n",
      "✅ Processed 4570/4900\n",
      "✅ Processed 4580/4900\n",
      "✅ Processed 4590/4900\n",
      "✅ Processed 4600/4900\n",
      "💾 Saved checkpoint after 460 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 4610/4900\n",
      "✅ Processed 4620/4900\n",
      "✅ Processed 4630/4900\n",
      "✅ Processed 4640/4900\n",
      "✅ Processed 4650/4900\n",
      "✅ Processed 4660/4900\n",
      "✅ Processed 4670/4900\n",
      "✅ Processed 4680/4900\n",
      "✅ Processed 4690/4900\n",
      "✅ Processed 4700/4900\n",
      "💾 Saved checkpoint after 470 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 4710/4900\n",
      "✅ Processed 4720/4900\n",
      "✅ Processed 4730/4900\n",
      "✅ Processed 4740/4900\n",
      "✅ Processed 4750/4900\n",
      "✅ Processed 4760/4900\n",
      "✅ Processed 4770/4900\n",
      "✅ Processed 4780/4900\n",
      "✅ Processed 4790/4900\n",
      "✅ Processed 4800/4900\n",
      "💾 Saved checkpoint after 480 batches at scores_flash_sentiment.csv\n",
      "✅ Processed 4810/4900\n",
      "✅ Processed 4820/4900\n",
      "✅ Processed 4830/4900\n",
      "✅ Processed 4840/4900\n",
      "✅ Processed 4850/4900\n",
      "✅ Processed 4860/4900\n",
      "✅ Processed 4870/4900\n",
      "✅ Processed 4880/4900\n",
      "✅ Processed 4890/4900\n",
      "✅ Processed 4900/4900\n",
      "💾 Saved checkpoint after 490 batches at scores_flash_sentiment.csv\n",
      "🎉 Finished. Final results saved at scores_flash_sentiment.csv\n"
     ]
    }
   ],
   "execution_count": 26
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T21:26:17.705986Z",
     "start_time": "2025-09-16T21:26:17.669297Z"
    }
   },
   "cell_type": "code",
   "source": [
    "import google.generativeai as genai\n",
    "import os\n",
    "import time\n",
    "from dotenv import load_dotenv\n",
    "\n",
    "# Load environment variables\n",
    "load_dotenv()\n",
    "genai.configure(api_key=os.getenv(\"GOOGLE_API_KEY\"))\n",
    "\n",
    "model = genai.GenerativeModel(\"gemini-2.5-pro\")\n",
    "\n",
    "def batch_eval(all_pairs, disease_ls, batch_size=20, max_retries=5, sleep_time=2, save_every=25, save_path=\"scores_checkpoint.csv\"):\n",
    "    all_scores = []\n",
    "    batch_counter = 0  # track how many batches processed\n",
    "\n",
    "    for start in range(0, all_pairs.shape[0], batch_size):\n",
    "        end = min(start + batch_size, all_pairs.shape[0])\n",
    "        batch = all_pairs.iloc[start:end]\n",
    "\n",
    "        # Build base prompt\n",
    "# Build base prompt\n",
    "        prompt = (\n",
    "            \"You are evaluating individual sentences for sentiment consistency.\\n\\n\"\n",
    "            \"Scoring instructions:\\n\"\n",
    "            \"- Assign each sentence a score between 0 and 1, rounded to two decimal places.\\n\"\n",
    "            \"- Criteria: The sentence should clearly express the SPECIFIED sentiment, remain coherent, and avoid conflicting or opposite emotions.\\n\"\n",
    "            \"- Use the full 0–1 range: 1 = perfectly consistent with the sentiment and natural; 0 = completely unusable.\\n\"\n",
    "            \"- 0.5 is the threshold: any sentence with a score ≤ 0.5 should be dropped to prevent sentiment drift.\\n\\n\"\n",
    "            \"Output requirements:\\n\"\n",
    "            \"- Output only the scores, one per line, in the same order as the input cases.\\n\"\n",
    "            \"- Do not include explanations, text, or formatting other than the numeric scores.\\n\\n\"\n",
    "        )\n",
    "\n",
    "        for i, row in batch.iterrows():\n",
    "            sent, senti = row[\"sentence\"], row[\"sentiment\"]\n",
    "            prompt += f\"Case {i}:\\nSentiment: {senti}\\nSentence: {sent}\\n\"\n",
    "\n",
    "        # Retry loop\n",
    "        scores = []\n",
    "        for attempt in range(1, max_retries + 1):\n",
    "            try:\n",
    "                response = model.generate_content(prompt)\n",
    "                scores = response.text.strip().splitlines()\n",
    "\n",
    "                if len(scores) == len(batch):\n",
    "                    break  # ✅ got the right number of outputs\n",
    "                else:\n",
    "                    print(\n",
    "                        f\"⚠️ Attempt {attempt}: Expected {len(batch)} scores, got {len(scores)}. Retrying...\"\n",
    "                    )\n",
    "                    time.sleep(sleep_time)\n",
    "\n",
    "            except Exception as e:\n",
    "                print(f\"❌ Error on attempt {attempt}: {e}\")\n",
    "                time.sleep(sleep_time)\n",
    "\n",
    "        if len(scores) != len(batch):\n",
    "            raise ValueError(\n",
    "                f\"Failed after {max_retries} retries: Expected {len(batch)} scores, got {len(scores)}\"\n",
    "            )\n",
    "\n",
    "        all_scores.extend(scores)\n",
    "        batch_counter += 1\n",
    "        print(f\"✅ Processed {end}/{all_pairs.shape[0]}\")\n",
    "\n",
    "        if batch_counter % save_every == 0:\n",
    "            pd.DataFrame({\"score\": all_scores}).to_csv(save_path, index=False)\n",
    "            print(f\"💾 Saved checkpoint after {batch_counter} batches at {save_path}\")\n",
    "\n",
    "    pd.DataFrame({\"score\": all_scores}).to_csv(save_path, index=False)\n",
    "    print(f\"🎉 Finished. Final results saved at {save_path}\")\n",
    "\n",
    "    return all_scores"
   ],
   "id": "9665c533cf2d421b",
   "outputs": [],
   "execution_count": 27
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T21:38:20.292121Z",
     "start_time": "2025-09-16T21:26:32.043161Z"
    }
   },
   "cell_type": "code",
   "source": "scores = batch_eval(df, emo_ls, batch_size=10, save_path=\"scores_pro_sentiment.csv\", save_every = 10)",
   "id": "7198c7b90efaa559",
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✅ Processed 10/500\n",
      "✅ Processed 20/500\n",
      "✅ Processed 30/500\n",
      "✅ Processed 40/500\n",
      "✅ Processed 50/500\n",
      "✅ Processed 60/500\n",
      "✅ Processed 70/500\n",
      "✅ Processed 80/500\n",
      "✅ Processed 90/500\n",
      "✅ Processed 100/500\n",
      "💾 Saved checkpoint after 10 batches at scores_pro_sentiment.csv\n",
      "✅ Processed 110/500\n",
      "✅ Processed 120/500\n",
      "✅ Processed 130/500\n",
      "✅ Processed 140/500\n",
      "✅ Processed 150/500\n",
      "✅ Processed 160/500\n",
      "✅ Processed 170/500\n",
      "✅ Processed 180/500\n",
      "✅ Processed 190/500\n",
      "✅ Processed 200/500\n",
      "💾 Saved checkpoint after 20 batches at scores_pro_sentiment.csv\n",
      "✅ Processed 210/500\n",
      "✅ Processed 220/500\n",
      "✅ Processed 230/500\n",
      "✅ Processed 240/500\n",
      "✅ Processed 250/500\n",
      "✅ Processed 260/500\n",
      "✅ Processed 270/500\n",
      "✅ Processed 280/500\n",
      "✅ Processed 290/500\n",
      "✅ Processed 300/500\n",
      "💾 Saved checkpoint after 30 batches at scores_pro_sentiment.csv\n",
      "✅ Processed 310/500\n",
      "✅ Processed 320/500\n",
      "✅ Processed 330/500\n",
      "✅ Processed 340/500\n",
      "✅ Processed 350/500\n",
      "✅ Processed 360/500\n",
      "✅ Processed 370/500\n",
      "✅ Processed 380/500\n",
      "✅ Processed 390/500\n",
      "✅ Processed 400/500\n",
      "💾 Saved checkpoint after 40 batches at scores_pro_sentiment.csv\n",
      "✅ Processed 410/500\n",
      "✅ Processed 420/500\n",
      "✅ Processed 430/500\n",
      "✅ Processed 440/500\n",
      "✅ Processed 450/500\n",
      "✅ Processed 460/500\n",
      "✅ Processed 470/500\n",
      "✅ Processed 480/500\n",
      "✅ Processed 490/500\n",
      "✅ Processed 500/500\n",
      "💾 Saved checkpoint after 50 batches at scores_pro_sentiment.csv\n",
      "🎉 Finished. Final results saved at scores_pro_sentiment.csv\n"
     ]
    }
   ],
   "execution_count": 28
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T21:39:39.072201Z",
     "start_time": "2025-09-16T21:39:39.007266Z"
    }
   },
   "cell_type": "code",
   "source": [
    "scores_flash =  pd.read_csv(\"scores_flash_sentiment.csv\")\n",
    "scores_pro = pd.read_csv(\"scores_pro_sentiment.csv\")"
   ],
   "id": "21a47039fee418fe",
   "outputs": [],
   "execution_count": 29
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T21:39:48.324993Z",
     "start_time": "2025-09-16T21:39:48.307179Z"
    }
   },
   "cell_type": "code",
   "source": "df.head(6)",
   "id": "7f0c961938041b24",
   "outputs": [
    {
     "data": {
      "text/plain": [
       "                                            sentence sentiment\n",
       "0  Though I only had the chance to experience the...  surprise\n",
       "1  The design and atmosphere managed to capture m...  surprise\n",
       "2  I found myself particularly drawn to the level...  surprise\n",
       "3  Even in that brief preview, the game showed no...  surprise\n",
       "4  I'm now eager to see how the full experience w...  surprise\n",
       "5  It's as if beneath the surface, turmoil stirs ...      fear"
      ],
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>sentence</th>\n",
       "      <th>sentiment</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>Though I only had the chance to experience the...</td>\n",
       "      <td>surprise</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>The design and atmosphere managed to capture m...</td>\n",
       "      <td>surprise</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>I found myself particularly drawn to the level...</td>\n",
       "      <td>surprise</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>Even in that brief preview, the game showed no...</td>\n",
       "      <td>surprise</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>I'm now eager to see how the full experience w...</td>\n",
       "      <td>surprise</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>It's as if beneath the surface, turmoil stirs ...</td>\n",
       "      <td>fear</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "execution_count": 30
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T21:39:58.231442Z",
     "start_time": "2025-09-16T21:39:58.217966Z"
    }
   },
   "cell_type": "code",
   "source": [
    "df[\"pro-score\"] = -1\n",
    "df[\"flash-score\"] = scores_flash.values\n",
    "df.iloc[:500, 2] = scores_pro.values.ravel()"
   ],
   "id": "3ff9c9d58360e4a9",
   "outputs": [],
   "execution_count": 31
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T21:40:01.932060Z",
     "start_time": "2025-09-16T21:40:01.782044Z"
    }
   },
   "cell_type": "code",
   "source": [
    "import matplotlib.pyplot as plt\n",
    "plt.scatter(df[\"pro-score\"].iloc[:500], df[\"flash-score\"].iloc[:500])"
   ],
   "id": "b9d0685927d77eef",
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x1701b0b80>"
      ]
     },
     "execution_count": 32,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ],
      "image/png": "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"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "execution_count": 32
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T21:40:18.221425Z",
     "start_time": "2025-09-16T21:40:18.189836Z"
    }
   },
   "cell_type": "code",
   "source": "df.to_csv(\"data_extension_sentiment.csv\", index = False)",
   "id": "1401a7bb09df4782",
   "outputs": [],
   "execution_count": 33
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 2
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython2",
   "version": "2.7.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
