{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from mr_eval.utils.utils import *\n",
    "import os\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "from matplotlib import font_manager\n",
    "\n",
    "def list_jsonl_files(folder_path):\n",
    "    \"\"\"\n",
    "    列举文件夹中的所有 .jsonl 文件\n",
    "    Args:\n",
    "        folder_path (str): 文件夹路径\n",
    "    Returns:\n",
    "        List[str]: 所有 .jsonl 文件的路径\n",
    "    \"\"\"\n",
    "    return [f for f in os.listdir(folder_path) if f.endswith(\".jsonl\")]\n",
    "## Model names\n",
    "## Model names\n",
    "prm_model_name_dict = dict(\n",
    "    skyworkprm_1_5B=\"Skywork-1.5B\",\n",
    "    skyworkprm_7B=\"Skywork-PRM-7B\",\n",
    "    llemma7b_prm_prm800k=\"Llemma-PRM800k-7B\",\n",
    "    llemma7b_prm_metamath=\"Llemma-MetaMath-7B\",\n",
    "    llemma7b_oprm_prm800k=\"Llemma-oprm-7B\",\n",
    "    mathminos_mistral=\"MATHMinos-7B\",\n",
    "    mathshepherd=\"MathShepherd-7B\",\n",
    "    reasoneval7b=\"ReasonEval-7B\",\n",
    "    llama3_1_8b_prm_mistral=\"RLHFlow-PRM-Mistral-8B\",\n",
    "    llama3_1_8b_prm_deepseek=\"RLHFlow-PRM-Deepseek-8B\",\n",
    "    reasoneval34b=\"ReasonEval-34B\",\n",
    ")\n",
    "close_model_name_dict = dict(\n",
    "    gpt4o=\"GPT-4o\",\n",
    "    o1mini=\"o1-mini\",\n",
    "    o1preview=\"o1-preview\",\n",
    "    gemini_2_flash=\"Gemini-2.0-flash-exp\",\n",
    "    gemini_2_thinking=\"Gemini-thinking\",\n",
    ")\n",
    "    \n",
    "open_model_name_dict = dict(\n",
    "    qwen_qwq=\"QwQ-Preview-32B\",\n",
    ")\n",
    "all_model_name_dict = {**prm_model_name_dict, **close_model_name_dict, **open_model_name_dict}\n",
    "\n",
    "\n",
    "classification_name_dict = dict(\n",
    "    domain_inconsistency=\"DC.\",\n",
    "    redundency=\"NR.\",\n",
    "    multi_solutions=\"MS.\",\n",
    "    deception=\"DR.\",\n",
    "    confidence=\"CI.\",\n",
    "    step_contradiction=\"SC.\",\n",
    "    circular=\"NCL.\",\n",
    "    missing_condition=\"PS.\",\n",
    "    counterfactual=\"ES.\"\n",
    ")\n",
    "classification_parallel_dict = dict(\n",
    "    simplicity=dict(\n",
    "        redundency=\"NR.\",\n",
    "        circular=\"NCL.\",\n",
    "    ),\n",
    "    soundness=dict(\n",
    "        counterfactual=\"ES.\",\n",
    "        step_contradiction=\"SC.\",\n",
    "        domain_inconsistency=\"DC.\",\n",
    "        confidence=\"CI.\",\n",
    "    ),\n",
    "    sensitivity=dict(\n",
    "        missing_condition=\"PS.\",\n",
    "        deception=\"DR.\",\n",
    "        multi_solutions=\"MS.\",\n",
    "    )\n",
    ")\n",
    "classifications = [\"redundency\", \"circular\", \"counterfactual\", \"step_contradiction\", \"domain_inconsistency\",  \"confidence\", \"missing_condition\", \"deception\", \"multi_solutions\", ]\n",
    "metrics = [\"f1\", \"negative_f1\", \"total_step_acc\", \"correct_step_acc\", \"wrong_step_acc\", \"first_error_acc\", \"similarity\",]\n",
    "\n",
    "## File paths\n",
    "res_dir = \"/mnt/petrelfs/songmingyang/code/reasoning/MR_Hallucination/mr_eval/scripts/logs/prmtest_classified\"\n",
    "res_files = list_jsonl_files(res_dir)\n",
    "res_names = [f.split(\".\")[0] for f in res_files]\n",
    "res_paths = [os.path.join(res_dir, f) for f in res_files]\n",
    "file_dict = dict(zip(res_names, res_paths))\n",
    "res_dict = {k: process_jsonl(v)[-1] for k, v in file_dict.items()}\n",
    "# detailed_log_dict = {k:v[\"\"]}\n",
    "display_models = [\"reasoneval34b\",\"mathshepherd\",\"gpt4o\",\"gemini_2_thinking\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'correct_step_acc': 1.0,\n",
       " 'wrong_step_acc': -1,\n",
       " 'total_step_acc': 1.0,\n",
       " 'first_error_acc': None,\n",
       " 'model_response_acc': 1.0,\n",
       " 'f1_matrix': {'TP': 6, 'FP': 0, 'TN': 0, 'FN': 0},\n",
       " 'correct_step_acc_list': [1, 1, 1, 1, 1, 1],\n",
       " 'wrong_step_acc_list': [],\n",
       " 'total_step_acc_list': [1, 1, 1, 1, 1, 1],\n",
       " 'first_error_acc_list': [],\n",
       " 'model_response_acc_list': [1.0]}"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "a = res_dict[\"gpt4o\"][\"detailed_logs\"]\n",
    "a[0].keys( )\n",
    "a[0][\"results\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "step_acc_dict = {k:{} for k in display_models}\n",
    "cnt = 0\n",
    "for model_name in display_models:\n",
    "    detailed_log = res_dict[model_name][\"detailed_logs\"]\n",
    "    for log in detailed_log:\n",
    "\n",
    "        if  \"validitiy\" in log and not log[\"validitiy\"]:\n",
    "            cnt += 1\n",
    "            continue\n",
    "        # if not log[\"prediction\"]  or (\"validity\" in log and not log[\"validitiy\"]) or  (\"validity\" in log[\"prediction\"] and not log[\"prediction\"][\"validity\"]):\n",
    "        #     cnt += 1\n",
    "        #     continue\n",
    "        error_steps = log[\"error_steps\"]\n",
    "        wrong_step_acc_list = log[\"results\"][\"wrong_step_acc_list\"]    \n",
    "        for idx,(error_step, wrong_step_acc) in enumerate(zip(error_steps, wrong_step_acc_list)):\n",
    "            step_acc_dict[model_name][error_step] = step_acc_dict[model_name].get(error_step, []) + [wrong_step_acc]\n",
    "\n",
    "for model_name, step_acc in step_acc_dict.items():\n",
    "    for k,v in step_acc.items():\n",
    "        step_acc_dict[model_name][k] = np.mean(v)\n",
    "\n",
    "step_acc_list = {}\n",
    "for model_name, step_acc in step_acc_dict.items():\n",
    "    step_acc_list[model_name] = [step_acc.get(i, 0)*100 for i in range(1, 50)]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def generate_sequence(num):\n",
    "    i = -num/2\n",
    "    while i < num:\n",
    "        yield i\n",
    "        i = i+1\n",
    "        \n",
    "def draw():\n",
    "    colors = [ \"#345a86\",\"#6cb4c7\",\"#fec869\",\"#e25f4f\",\"#40724e\", '#2691b1','#fab94f', '#e05e4e','#ff968d', '#b2ff89','#bc9eff','#bdbca5', 'grey', 'yellow','k','brown','grey','cyan',\"pink\"]\n",
    "    draw_steps = list(range(2, 11))\n",
    "    # for step in draw_steps:\n",
    "    plt.figure(figsize=(6,4))\n",
    "    generator = generate_sequence(len(display_models))\n",
    "    for model_name in display_models:\n",
    "        bar_width=0.7/len(display_models)\n",
    "        index=np.arange(len(draw_steps))\n",
    "        \n",
    "        font_path = '/mnt/petrelfs/songmingyang/code/tools/tool_code/result_dealing/draw_fig/Comic Sans MS.ttf'  \n",
    "        font_path = \"/mnt/petrelfs/songmingyang/.config/fonts/Times New Roman.ttf\"\n",
    "        custom_font = font_manager.FontProperties(fname=font_path,size=14)\n",
    "        custom_font_small = font_manager.FontProperties(fname=font_path)\n",
    "        plt.grid(True,axis='y',zorder=1)\n",
    "        plt.bar(index+next(generator)*bar_width,step_acc_list[model_name][draw_steps[0]:draw_steps[-1]+1],bar_width,label=f'{all_model_name_dict[model_name]}',color=colors.pop(0),zorder=3, edgecolor='black', linewidth=0.5)\n",
    "        # plt.bar(index+next(generator)*bar_width,base_data,bar_width,label=f'LLaVA V1.5',color=colors.pop(0),zorder=3, edgecolor='black')\n",
    "        \n",
    "        plt.xticks(index,draw_steps,fontproperties=custom_font)\n",
    "        plt.yticks(fontproperties=custom_font)\n",
    "        plt.ylabel(\"Accuracy(%)\",fontproperties=custom_font)\n",
    "        plt.xlabel(\"Step\",fontproperties=custom_font)\n",
    "        plt.ylim(0, 75)\n",
    "        # plt.legend(loc='lower left',zorder=100)\n",
    "        plt.legend(loc='upper center', ncol=4,prop=custom_font_small, bbox_to_anchor=(0.5, -0.14),frameon=False)\n",
    "        # 移除子图的边框\n",
    "        ax = plt.gca()\n",
    "        # 隐藏右边框和上边框\n",
    "        ax.spines['right'].set_visible(False)\n",
    "        ax.spines['top'].set_visible(False)\n",
    "        ax.spines['left'].set_visible(False)\n",
    "        # ax.spines['bottom'].set_color('red')\n",
    "        ax.spines['bottom'].set_linewidth(2)\n",
    "        \n",
    "        # endoffig=7.5\n",
    "        # begin=3.4\n",
    "        # plt.xlim(-1,endoffig)\n",
    "        # end=endoffig\n",
    "        # arrow_center=(begin+end)/2\n",
    "        # height=70\n",
    "        \n",
    "        # plt.axvline(x=begin, color='r', linestyle='--')\n",
    "        # plt.axvline(x=end, color='r', linestyle='--')\n",
    "        \n",
    "        # plt.annotate('', xy=(begin, height), xytext=(end, height), \n",
    "        #     arrowprops=dict(arrowstyle=\"<->\", color='blue'),\n",
    "        #     ha='center', va='bottom',fontproperties=custom_font)\n",
    "        # plt.axvspan(begin,end , facecolor='y', alpha=0.5)\n",
    "        # plt.annotate('Low Resource Languages', xy=(arrow_center, 95), xycoords='data',\n",
    "        #     xytext=(0, -10), textcoords='offset points',\n",
    "        #     ha='center', va='bottom',fontproperties=custom_font,\n",
    "        #     bbox=dict(boxstyle='round,pad=0.5', facecolor=\"none\", edgecolor=\"none\", lw=0.5))\n",
    "        # plt.savefig(\"./res/preliminary_dpo.pdf\", format='pdf')\n",
    "        \n",
    "        # plt.annotate('', xy=(-1, height), xytext=(begin, height), \n",
    "        #     arrowprops=dict(arrowstyle=\"<->\", color='blue'),\n",
    "        #     ha='center', va='bottom',fontproperties=custom_font)\n",
    "        # plt.axvspan(-1,begin-0.01 , facecolor='pink', alpha=0.5)\n",
    "        # plt.annotate('High Resource Languages', xy=((begin-1)/2, 91), xycoords='data',\n",
    "        #     xytext=(0, 0), textcoords='offset points',\n",
    "        #     ha='center', va='bottom',fontproperties=custom_font,\n",
    "        #     bbox=dict(boxstyle='round,pad=0.5', facecolor=\"none\", edgecolor=\"none\", lw=0.5))\n",
    "        \n",
    "        \n",
    "        # plt.annotate('', xy=(end, 0.85), xytext=(endoffig, 0.85), \n",
    "        #  arrowprops=dict(arrowstyle=\"<->\", color='blue'),\n",
    "        #  ha='center', va='bottom',fontproperties=custom_font)\n",
    "        # plt.axvspan(end,endoffig , facecolor='gray', alpha=0.5)\n",
    "        # plt.annotate('LRL', xy=((end+endoffig)/2, 0.86), xycoords='data',\n",
    "        #  xytext=(0, 0), textcoords='offset points',\n",
    "        #  ha='center', va='bottom',fontproperties=custom_font,\n",
    "        #  bbox=dict(boxstyle='round,pad=0.5', facecolor=\"none\", edgecolor=\"none\", lw=0.5))\n",
    "        # plt.savefig(\"./res/preliminary_dpo.pdf\", format='pdf')\n",
    "        # for spine in ax.spines.values():\n",
    "        #     spine.set_visible(False)\n",
    "        # plt.title(f'{dataset_res[\"dataset_name\"]}_{type_item[\"type_name\"]}')\n",
    "\n",
    "        # a_1000 = sum(plot_data[\"sft_1000\"][\"accuracy\"]) / len(plot_data[\"sft_1000\"][\"accuracy\"])\n",
    "        # d_sh_hp = sum(plot_data[\"dpo_sh_hp\"][\"accuracy\"]) / len(plot_data[\"dpo_sh_hp\"][\"accuracy\"])\n",
    "        # print(f\"d_sh_hp: {d_sh_hp}, a_1000: {a_1000}\")\n",
    "        ax.annotate('', xy=(0, 1.05), xycoords='axes fraction', xytext=(0, -0.01),\n",
    "            arrowprops=dict(arrowstyle=\"->\", color='black',linewidth=2))\n",
    "        plt.savefig(\"./res/step_acc.pdf\", format='pdf',bbox_inches='tight')\n",
    "draw()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "smoe",
   "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.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
