{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import os, json, time, threading, random\n",
    "from typing import Optional, Tuple, List, Dict\n",
    "from concurrent.futures import ThreadPoolExecutor, as_completed\n",
    "\n",
    "import pandas as pd\n",
    "from openai import OpenAI"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "data = pd.read_csv('annotated_news_with_metrics.csv')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "SYSTEM_PROMPT_SPANS = (\n",
    "    \"You are a careful copy editor. Given a paragraph, extract the minimal set of short \"\n",
    "    \"verbatim spans (quoted) that are indicative of \\\"slop\\\" according to the guide, then provide a brief reasoning.\\n\"\n",
    "    \"The guide is provided below. “Slop” refers to AI-generated text that is low-quality. It can appear superficially correct but is some combination of generic, overly verbose, inaccurate, irrelevant to its intended purpose, and contributing little meaningful value to the reader (despite sounding fluent). “Slop” typically displays patterns of repetition, formulaic structure, vague language, and an absence of authentic perspective.\"\n",
    "    \"Factuality: Incorrect or fabricated information, Misleading or fallacious claims. Example: “Dr. Sarah Johnson of Harvard University published groundbreaking research on this topic in 2022.” (Slop if Dr. Johnson doesn’t exist, isn’t at Harvard, or didn’t publish such research)\"\n",
    "    \"Bias: Lack of appropriate perspective or over-standardization. Example: “The economic policy changes of 2023 were universally beneficial.” (Slop because it presents a one-sided view of complex policy impacts)\"\n",
    "    \"Information Density: Text that is verbose but conveys little actual information. Excessive filler words. Example: “In today’s fast-paced modern world of cutting-edge technology and innovation, it has become increasingly important to consider the various factors and elements that contribute to our understanding of this complex and multifaceted issue.” (Slop because it uses many words to say almost nothing)\"\n",
    "    \"Information Relevance: Appropriateness to the specific context, query, or task. For text with no additional context (e.g., an article), consider internal relevance within the passage. Example: In response to “How can I improve my marathon time?”: “Running is an excellent form of exercise with many health benefits including improved cardiovascular function, enhanced mood, and weight management.” (Slop because it doesn’t address the specific question about improving marathon times)\"\n",
    "    \"Repetition:  Excessive use of the same words or phrases. Low diversity in vocabulary and expression. Example: “The project was a success. The team accomplished their goals successfully. The successful outcome was due to the team’s hard work.” (Slop due to repetition of “success/successful” without adding new information)\"\n",
    "    \"Templatedness: Over-reliance on formulaic structures and patterns. Predictable formatting patterns (e.g., excessive use of bullet points). Frequent appearance of text that follows a common pattern (e.g., “Mr. X, a Y-year-old Z”). Example: “Dr. Smith, a researcher at Oxford University, found that… Professor Johnson, a scientist at Cambridge University, discovered that… Dr. Williams, an expert at Yale University, confirmed that…” (Slop because it follows the same formula repeatedly)\" \n",
    "    \"Coherence: Poor sentence structure or organization. Text that requires significant effort to follow. Example: “Climate change is affecting global temperatures. Polar bears are mammals. Ice cream melts in warm weather. Arctic ice is melting. Some people enjoy winter sports.” (Slop because the sentences, while related to temperature, don’t flow logically)\"\n",
    "    \"Fluency: Strange turns of phrases or unnatural language. Example: “The earthen area that formerly held the puddle was now dry.” (Slop because natural language would simply say “The puddle had dried up” or “The ground where the puddle had been was now dry”)\" \n",
    "    \"Word Complexity: Unnecessary jargon or complicated terminology. Overuse of rare words. Example: In a general article about gardening: “The phenolic compounds in certain cultivars exhibit antimicrobial properties that mitigate pathogenic microorganism colonization.” (Slop because it uses unnecessarily complex terminology for the intended audience)\" \n",
    "    \"Tone: Appropriate voice and style for the context. Example: In a blog post about personal travel experiences: “The aforementioned destination offers numerous recreational activities for tourists. Visitors may engage in swimming, hiking, or dining at local establishments.” (Slop because it uses an inappropriately formal tone for a personal blog)\"\n",
    "    \n",
    "    'Return a JSON object: { \"spans\": [\"...\",\"...\"], \"reasoning\": \"...\" }'\n",
    ")\n",
    "\n",
    "SYSTEM_PROMPT_LABEL = (\n",
    "    \"You are a careful copy editor. Given a piece of text, return a binary assessment of whether this is overall slop. \"\n",
    "    \"The guide is provided below. “Slop” refers to AI-generated text that is low-quality. It can appear superficially correct but is some combination of generic, overly verbose, inaccurate, irrelevant to its intended purpose, and contributing little meaningful value to the reader (despite sounding fluent). “Slop” typically displays patterns of repetition, formulaic structure, vague language, and an absence of authentic perspective.\"\n",
    "    \"Factuality: Incorrect or fabricated information, Misleading or fallacious claims. Example: “Dr. Sarah Johnson of Harvard University published groundbreaking research on this topic in 2022.” (Slop if Dr. Johnson doesn’t exist, isn’t at Harvard, or didn’t publish such research)\"\n",
    "    \"Bias: Lack of appropriate perspective or over-standardization. Example: “The economic policy changes of 2023 were universally beneficial.” (Slop because it presents a one-sided view of complex policy impacts)\"\n",
    "    \"Information Density: Text that is verbose but conveys little actual information. Excessive filler words. Example: “In today’s fast-paced modern world of cutting-edge technology and innovation, it has become increasingly important to consider the various factors and elements that contribute to our understanding of this complex and multifaceted issue.” (Slop because it uses many words to say almost nothing)\"\n",
    "    \"Information Relevance: Appropriateness to the specific context, query, or task. For text with no additional context (e.g., an article), consider internal relevance within the passage. Example: In response to “How can I improve my marathon time?”: “Running is an excellent form of exercise with many health benefits including improved cardiovascular function, enhanced mood, and weight management.” (Slop because it doesn’t address the specific question about improving marathon times)\"\n",
    "    \"Repetition:  Excessive use of the same words or phrases. Low diversity in vocabulary and expression. Example: “The project was a success. The team accomplished their goals successfully. The successful outcome was due to the team’s hard work.” (Slop due to repetition of “success/successful” without adding new information)\"\n",
    "    \"Templatedness: Over-reliance on formulaic structures and patterns. Predictable formatting patterns (e.g., excessive use of bullet points). Frequent appearance of text that follows a common pattern (e.g., “Mr. X, a Y-year-old Z”). Example: “Dr. Smith, a researcher at Oxford University, found that… Professor Johnson, a scientist at Cambridge University, discovered that… Dr. Williams, an expert at Yale University, confirmed that…” (Slop because it follows the same formula repeatedly)\" \n",
    "    \"Coherence: Poor sentence structure or organization. Text that requires significant effort to follow. Example: “Climate change is affecting global temperatures. Polar bears are mammals. Ice cream melts in warm weather. Arctic ice is melting. Some people enjoy winter sports.” (Slop because the sentences, while related to temperature, don’t flow logically)\"\n",
    "    \"Fluency: Strange turns of phrases or unnatural language. Example: “The earthen area that formerly held the puddle was now dry.” (Slop because natural language would simply say “The puddle had dried up” or “The ground where the puddle had been was now dry”)\" \n",
    "    \"Word Complexity: Unnecessary jargon or complicated terminology. Overuse of rare words. Example: In a general article about gardening: “The phenolic compounds in certain cultivars exhibit antimicrobial properties that mitigate pathogenic microorganism colonization.” (Slop because it uses unnecessarily complex terminology for the intended audience)\" \n",
    "    \"Tone: Appropriate voice and style for the context. Example: In a blog post about personal travel experiences: “The aforementioned destination offers numerous recreational activities for tourists. Visitors may engage in swimming, hiking, or dining at local establishments.” (Slop because it uses an inappropriately formal tone for a personal blog)\"\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from typing import List, Dict\n",
    "\n",
    "def get_text(t: str, sp: List[Dict[str, int | str]]):\n",
    "    if type(sp) == str: \n",
    "        sp = eval(sp)\n",
    "    extracted = []\n",
    "    label = None\n",
    "    for span in sp:\n",
    "        start, end = span[\"start\"], span[\"end\"]\n",
    "        extracted.append(t[start:end])\n",
    "        label = span[\"label\"] \n",
    "    \n",
    "    return {\"spans\": extracted, \"label\": label}\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "data['combined_spans_collapsed_text'] = data.apply(lambda x: get_text(x.text, x.combined_spans_collapsed), axis=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "data['prompt_spans'] = data['text'].apply(lambda x: SYSTEM_PROMPT_SPANS + \"\\n\\nPARAGRAPH: \" + x +'Task: List the minimal verbatim spans (as quoted substrings) and explain briefly. Format: { \"spans\": [\"...\"], \"reasoning\": \"...\" }') \n",
    "data_label_only['prompt_labels'] = data['text'].apply(lambda x: SYSTEM_PROMPT_LABEL + \"\\n\\TEXT: \" + x +'Task: Is this slop (0 = no, 1 = yes)') "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Zero-Shot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def gpt_batch_responses(\n",
    "    df\n",
    "    key_path,\n",
    "    model = \"gpt-5\",\n",
    "    prompt_col = 'prompt_spans',\n",
    "    max_completion_tokens = 1024,\n",
    "    n_workers = 8,\n",
    "    out_col = \"gpt5_response\",\n",
    "    checkpoint_path = \"gpt5_responses.jsonl\",\n",
    "    resume = True,\n",
    "    progress_every: int = 25,\n",
    "    id_col = None,\n",
    "    k = 0,        \n",
    "    shots_text_col = \"text\",\n",
    "    shots_label_col  = \"combined_spans_collapsed\",\n",
    "    shots_separator = \"\\n\\n\",\n",
    "    fewshot_seed = None,\n",
    "    sample_with_replacement = False,\n",
    "):\n",
    "    # openai key\n",
    "    with open(key_path, \"r\") as f:\n",
    "        api_key = f.read().strip()\n",
    "    if not api_key:\n",
    "        raise ValueError(\"API key file is empty.\")\n",
    "\n",
    "    # ID + prompt building\n",
    "    if id_col:\n",
    "        if id_col not in df.columns:\n",
    "            raise KeyError(f\"id_col '{id_col}' not found in df.\")\n",
    "        ids = df[id_col].tolist()\n",
    "    else:\n",
    "        ids = df.index.tolist()\n",
    "    \n",
    "    prompts = df[prompt_col].fillna(\"\").astype(str).tolist()\n",
    "    \n",
    "    if len(ids) != len(prompts):\n",
    "        raise ValueError(\"Length mismatch between ids and prompts.\")\n",
    "\n",
    "    # load checkpoint (if any)\n",
    "    done = {}\n",
    "    if resume and os.path.exists(checkpoint_path):\n",
    "        with open(checkpoint_path, \"r\") as f:\n",
    "            for line in f:\n",
    "                try:\n",
    "                    rec = json.loads(line)\n",
    "                    if \"response\" in rec:\n",
    "                        done[str(rec[\"row_id\"])] = rec[\"response\"] \n",
    "                except Exception:\n",
    "                    pass\n",
    "\n",
    "    # few shot set up \n",
    "    rng = random.Random(fewshot_seed)\n",
    "\n",
    "    # build few-shot context excluding row at position i \n",
    "    n_rows = len(df)\n",
    "    all_positions = list(range(n_rows))\n",
    "\n",
    "    # worklist (row_id, final_prompt) for rows not yet done\n",
    "    work = []\n",
    "    for pos, (rid, base_prompt) in enumerate(zip(ids, prompts)):\n",
    "        if str(rid) in done:\n",
    "            continue\n",
    "        context = build_fewshot_context(\n",
    "            exclude_pos=pos, \n",
    "            n_rows=n_rows, \n",
    "            all_positions=all_positions, \n",
    "            df=df, \n",
    "            k=k, \n",
    "            shots_text_col=shots_text_col, \n",
    "            shots_label_col=shots_label_col, \n",
    "            shots_separator=shots_separator, \n",
    "            rng=rng, \n",
    "            sample_with_replacement=sample_with_replacement\n",
    "        )\n",
    "        if context:\n",
    "            final_prompt = f\"Examples: {context}\\n\\n {base_prompt} \\n\\n \"\n",
    "        else:\n",
    "            final_prompt = base_prompt\n",
    "        work.append((rid, final_prompt))\n",
    "\n",
    "    total = len(ids)\n",
    "    already = len(done)\n",
    "    remaining = len(work)\n",
    "    start = time.time()\n",
    "    lock = threading.Lock()\n",
    "\n",
    "    # prepare output df, adds existing responses if resuming\n",
    "    out = df.copy()\n",
    "    out[out_col] = pd.NA\n",
    "    for rid, resp in done.items():\n",
    "        if id_col:\n",
    "            out.loc[out[id_col].astype(str) == rid, out_col] = resp\n",
    "        else:\n",
    "            if str(rid) in set(out.index.astype(str)):\n",
    "                out.at[out.index[out.index.astype(str) == rid][0], out_col] = resp\n",
    "\n",
    "    if remaining == 0:\n",
    "        print(f\"[resume] Nothing to do ({already}/{total} already complete).\")\n",
    "        return out\n",
    "\n",
    "    completed = already\n",
    "    \n",
    "    def _update_eta():\n",
    "        elapsed = time.time() - start\n",
    "        rate = completed / elapsed if elapsed > 0 else 0.0\n",
    "        rem = total - completed\n",
    "        eta = (rem / rate) if rate > 0 else float(\"inf\")\n",
    "        mm = int(eta // 60) if eta < 1e12 else 0\n",
    "        ss = int(eta % 60) if eta < 1e12 else 0\n",
    "        print(f\"[{completed}/{total}] {rate:.2f} req/s | ETA ~ {mm}m{ss}s\")\n",
    "\n",
    "    with ThreadPoolExecutor(max_workers=n_workers) as ex:\n",
    "        futures = {ex.submit(call_and_checkpoint, rid, prmpt): str(rid) for rid, prmpt in work}\n",
    "        for i, fu in enumerate(as_completed(futures), 1):\n",
    "            rid = futures[fu]\n",
    "            text, err = fu.result()\n",
    "            if id_col:\n",
    "                out.loc[out[id_col].astype(str) == rid, out_col] = text if text is not None else f\"[ERROR] {err}\"\n",
    "            else:\n",
    "                mask = out.index.astype(str) == rid\n",
    "                out.loc[mask, out_col] = text if text is not None else f\"[ERROR] {err}\"\n",
    "            completed += 1\n",
    "            if (completed % progress_every == 0) or (completed == total):\n",
    "                _update_eta()\n",
    "\n",
    "    return out\n",
    "\n",
    "def build_fewshot_context(exclude_pos, \n",
    "                          n_rows, \n",
    "                          all_positions, \n",
    "                          df, \n",
    "                          k, \n",
    "                          shots_text_col, \n",
    "                          shots_label_col, \n",
    "                          shots_separator, \n",
    "                          rng, \n",
    "                          sample_with_replacement):\n",
    "        if k <= 0 or n_rows <= 1:\n",
    "            return \"\"\n",
    "        \n",
    "        # pool excluding current row\n",
    "        pool = all_positions[:exclude_pos] + all_positions[exclude_pos+1:]\n",
    "        if not sample_with_replacement:\n",
    "            _k = min(k, len(pool))\n",
    "            shots_idx = rng.sample(pool, _k)\n",
    "        else:\n",
    "            # sample with replacement (even if pool is small)\n",
    "            shots_idx = [rng.choice(pool) for _ in range(k)]\n",
    "            \n",
    "        shots = []\n",
    "        \n",
    "        for j in shots_idx:\n",
    "            t = str(df.iloc[j][shots_text_col])\n",
    "            y = str(df.iloc[j][shots_label_col])\n",
    "            shots.append(f\"{t}\\n{y}\")\n",
    "            \n",
    "        return shots_separator.join(shots)\n",
    "\n",
    "def call_and_checkpoint(row_id, prompt):\n",
    "    client = OpenAI(api_key=api_key)\n",
    "    tries, backoff = 0, 1.0\n",
    "    while True:\n",
    "        try:\n",
    "            msgs = ([{\"role\":\"system\",\"content\":system}] if system else []) + [{\"role\":\"user\",\"content\":prompt}]\n",
    "            r = client.responses.create(\n",
    "                model=model,\n",
    "                input=msgs,\n",
    "                # temperature=temperature,\n",
    "                # max_output_tokens=max_completion_tokens,\n",
    "            )\n",
    "            # print(msgs)\n",
    "            try:\n",
    "                text = r.output_text\n",
    "            except Exception:\n",
    "                text = \"\"\n",
    "                if hasattr(r, \"output\"):\n",
    "                    parts = []\n",
    "                    for block in getattr(r, \"output\", []):\n",
    "                        for c in getattr(block, \"content\", []):\n",
    "                            if getattr(c, \"type\", None) in (\"output_text\",\"text\"):\n",
    "                                parts.append(getattr(c, \"text\", \"\"))\n",
    "                    text = \"\".join(parts) if parts else str(r)\n",
    "                else:\n",
    "                    text = str(r)\n",
    "            rec = {\"row_id\": row_id, \"response\": text}\n",
    "            with lock:\n",
    "                with open(checkpoint_path, \"a\") as f:\n",
    "                    f.write(json.dumps(rec) + \"\\n\")\n",
    "            return text, None\n",
    "        except Exception as e:\n",
    "            tries += 1\n",
    "            if tries >= 5:\n",
    "                rec = {\"row_id\": row_id, \"error\": f\"{type(e).__name__}: {e}\"}\n",
    "                with lock:\n",
    "                    with open(checkpoint_path, \"a\") as f:\n",
    "                        f.write(json.dumps(rec) + \"\\n\")\n",
    "                return None, e\n",
    "            time.sleep(min(60.0, backoff))\n",
    "            backoff *= 2.0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "from typing import List, Dict, Optional\n",
    "import pandas as pd\n",
    "\n",
    "def run_fewshot_batches_with_prompt(\n",
    "    df, \n",
    "    prompt_str,\n",
    "    key_path,\n",
    "    ks = (3, 5),\n",
    "    base_name = \"run\",\n",
    "    out_dir = \"fewshot_outputs\",\n",
    "    model = \"gpt-5\",\n",
    "    max_completion_tokens = 1024,\n",
    "    temperature = 0.2,\n",
    "    n_workers = 8,\n",
    "    resume = True,\n",
    "    id_col = None,\n",
    "    out_format = \"csv\",  # \"csv\" or \"parquet\"\n",
    "    fewshot_seed = 42,\n",
    "    shots_text_col = \"text\",\n",
    "    shots_label_col = \"combined_spans_collapsed_text\",\n",
    "    sample_with_replacement = False,\n",
    "):\n",
    "    # adds a constant crafted prompt into df and runs gpt_batch_responses for each k in ks\n",
    "    os.makedirs(out_dir, exist_ok=True)\n",
    "    saved: Dict[int, str] = {}\n",
    "\n",
    "    for k in ks:\n",
    "        out_col = f\"gpt5_response_k{k}\"\n",
    "        checkpoint_path = os.path.join(out_dir, f\"{base_name}.k{k}.jsonl\")\n",
    "        save_path = os.path.join(out_dir, f\"{base_name}.k{k}.{out_format}\")\n",
    "\n",
    "        df_out = gpt5_batch_responses_checkpointed(\n",
    "            df=df,\n",
    "            prompt_col=\"prompt_spans\",\n",
    "            key_path=key_path,\n",
    "            model=model,\n",
    "            system=system,\n",
    "            max_completion_tokens=max_completion_tokens,\n",
    "            temperature=temperature,\n",
    "            n_workers=n_workers,\n",
    "            out_col=out_col,\n",
    "            checkpoint_path=checkpoint_path,\n",
    "            resume=resume,\n",
    "            id_col=id_col,\n",
    "            # few-shot params\n",
    "            k=k,\n",
    "            shots_text_col=shots_text_col,\n",
    "            shots_label_col=shots_label_col,\n",
    "            fewshot_seed=fewshot_seed,\n",
    "            sample_with_replacement=sample_with_replacement,\n",
    "        )\n",
    "\n",
    "        if out_format.lower() == \"csv\":\n",
    "            df_out.to_csv(save_path, index=False)\n",
    "        elif out_format.lower() == \"parquet\":\n",
    "            df_out.to_parquet(save_path, index=False)\n",
    "        else:\n",
    "            raise ValueError(\"out_format must be 'csv' or 'parquet'\")\n",
    "\n",
    "        saved[k] = save_path\n",
    "        print(f\"[saved] k={k} to {save_path}\")\n",
    "\n",
    "    return saved\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[25/2621] 0.35 req/s | ETA ~ 122m48s\n",
      "[50/2621] 0.36 req/s | ETA ~ 117m59s\n",
      "[75/2621] 0.39 req/s | ETA ~ 108m59s\n",
      "[100/2621] 0.38 req/s | ETA ~ 110m49s\n",
      "[125/2621] 0.38 req/s | ETA ~ 110m49s\n",
      "[150/2621] 0.38 req/s | ETA ~ 108m45s\n",
      "[175/2621] 0.39 req/s | ETA ~ 105m15s\n",
      "[200/2621] 0.39 req/s | ETA ~ 104m22s\n",
      "[225/2621] 0.40 req/s | ETA ~ 101m3s\n",
      "[250/2621] 0.40 req/s | ETA ~ 98m38s\n",
      "[275/2621] 0.39 req/s | ETA ~ 100m40s\n",
      "[300/2621] 0.39 req/s | ETA ~ 99m47s\n",
      "[325/2621] 0.39 req/s | ETA ~ 97m17s\n",
      "[350/2621] 0.39 req/s | ETA ~ 95m57s\n",
      "[375/2621] 0.40 req/s | ETA ~ 94m33s\n",
      "[400/2621] 0.40 req/s | ETA ~ 91m30s\n"
     ]
    }
   ],
   "source": [
    "paths = run_fewshot_batches_with_prompt(\n",
    "    df=data,                                # has columns \"text\" and \"combined_spans_collapsed\"\n",
    "    prompt_str=SYSTEM_PROMPT_SPANS,\n",
    "    key_path=\".openai_key.txt\",\n",
    "    ks=[1, 3, 5],\n",
    "    base_name=\"slop_eval\",\n",
    "    out_dir=\"fewshot_outputs\",\n",
    "    model=\"gpt-5\",\n",
    "    system=\"You are a careful annotator. Produce concise labels.\",\n",
    "    shots_text_col=\"text\",\n",
    "    shots_label_col=\"combined_spans_collapsed\",\n",
    "    fewshot_seed=123,\n",
    ")\n",
    "\n",
    "print(paths)\n",
    "\n",
    "# {1: 'fewshot_outputs/slop_eval.k1.csv', 3: 'fewshot_outputs/slop_eval.k3.csv', 5: 'fewshot_outputs/slop_eval.k5.csv'}\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
