{
 "cells": [
  {
   "cell_type": "code",
   "outputs": [],
   "source": [
    "# For data manipulation and analysis\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "\n",
    "# For text preprocessing\n",
    "import re\n",
    "import nltk\n",
    "from nltk.corpus import stopwords\n",
    "from nltk.stem import WordNetLemmatizer\n",
    "from sklearn.feature_extraction.text import TfidfVectorizer\n",
    "import datetime\n",
    "import string\n",
    "\n",
    "# For multilabel classification\n",
    "from sklearn.preprocessing import MultiLabelBinarizer\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "from sklearn.multiclass import OneVsRestClassifier\n",
    "from sklearn.ensemble import RandomForestClassifier\n",
    "from sklearn.naive_bayes import MultinomialNB\n",
    "import os\n",
    "\n",
    "# For neural networks\n",
    "\n",
    "\n",
    "\n",
    "# For model evaluation\n",
    "from sklearn.metrics import accuracy_score, f1_score, precision_score, recall_score, classification_report, confusion_matrix"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2024-09-03T08:38:24.278847Z",
     "start_time": "2024-09-03T08:38:23.957830Z"
    }
   },
   "id": "255a046e05404536",
   "execution_count": 1
  },
  {
   "cell_type": "code",
   "outputs": [],
   "source": [
    "# clusters, tags, sentiment\n",
    "tags_c = pd.read_csv('../dataset/final_clusters.csv')\n",
    "tags = pd.read_csv(\"../dataset/tags_withglovevec.csv\")\n",
    "df_s = pd.read_csv('../dataset/df_tag_sentiment.csv')\n"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2024-09-03T08:38:24.761341Z",
     "start_time": "2024-09-03T08:38:24.279589Z"
    }
   },
   "id": "7950fef131ced877",
   "execution_count": 2
  },
  {
   "cell_type": "markdown",
   "source": [
    "# By Fold Evaluation Results"
   ],
   "metadata": {
    "collapsed": false
   },
   "id": "dd92ba665e67a49f"
  },
  {
   "cell_type": "code",
   "outputs": [],
   "source": [
    "# calculation of similarity\n",
    "def cosine_similarity(v1, v2):\n",
    "    '''Cosine similarity function only computes similarity for NON-NAN values'''\n",
    "    # Indices where both v1 and v2 are not NaN\n",
    "    shared_idx = np.where(~np.isnan(v1) & ~np.isnan(v2))\n",
    "\n",
    "    # If no shared indices, return 0\n",
    "    if len(shared_idx[0]) == 0:\n",
    "        return 0\n",
    "\n",
    "    # Extract shared values\n",
    "    v1_shared = v1[shared_idx]\n",
    "    v2_shared = v2[shared_idx]\n",
    "\n",
    "    # Compute the dot product and norms only on shared values\n",
    "    dot_product = np.dot(v1_shared, v2_shared)\n",
    "    norm_v1 = np.linalg.norm(v1_shared)\n",
    "    norm_v2 = np.linalg.norm(v2_shared)\n",
    "\n",
    "    # Prevent division by zero\n",
    "    if norm_v1 == 0 or norm_v2 == 0:\n",
    "        return 0\n",
    "\n",
    "    return dot_product / (norm_v1 * norm_v2)\n",
    "\n",
    "\n",
    "def get_sentiment_label(sentiment):\n",
    "    if sentiment > 0.5:\n",
    "        return \"positive\"\n",
    "    elif sentiment < -0.5:\n",
    "        return \"negative\"\n",
    "    else:\n",
    "        return \"neutral\"\n",
    "\n",
    "\n",
    "def get_dominant_sentiment(user_sentiments):\n",
    "    pos_count = sum(1 for s in user_sentiments if s > 0.5)\n",
    "    neg_count = sum(1 for s in user_sentiments if s < -0.5)\n",
    "    neutral_count = len(user_sentiments) - pos_count - neg_count\n",
    "\n",
    "    if pos_count > neg_count and pos_count > neutral_count:\n",
    "        return \"positive\"\n",
    "    elif neg_count > pos_count and neg_count > neutral_count:\n",
    "        return \"negative\"\n",
    "    else:\n",
    "        return \"neutral\"\n",
    "\n",
    "\n",
    "def get_k_nearest_neighbors(target_user, matrix, k):\n",
    "    similarities = {}\n",
    "\n",
    "    for user in matrix.index:\n",
    "        if user == target_user:\n",
    "            continue\n",
    "        common_movies = matrix.loc[target_user].dropna().index.intersection(matrix.loc[user].dropna().index)\n",
    "        if len(common_movies) > 0:\n",
    "            user_vector = matrix.loc[target_user][common_movies].values.reshape(1, -1)\n",
    "            target_vector = matrix.loc[user][common_movies].values.reshape(1, -1)\n",
    "            sim = cosine_similarity(user_vector, target_vector)\n",
    "\n",
    "            similarities[user] = sim\n",
    "\n",
    "    # Sort by similarity\n",
    "    sorted_neighbors = sorted(similarities.items(), key=lambda x: x[1], reverse=True)\n",
    "    return sorted_neighbors[:k]\n",
    "\n",
    "\n",
    "def get_distributed_recommendations(target_user, matrix, new_to_original_index, df_sent, k):\n",
    "    # Get the sentiments for movies rated by the target user\n",
    "    user_sentiments = matrix.loc[target_user].dropna()\n",
    "\n",
    "    # Determine the dominant sentiment of the target user\n",
    "    dominant_sentiment = get_dominant_sentiment(user_sentiments)\n",
    "\n",
    "    # Get the k-most similar users\n",
    "    similar_users = get_k_nearest_neighbors(target_user, matrix, k)\n",
    "    sentiment_buckets = defaultdict(list)\n",
    "    # For each similar user, gather movies they've rated\n",
    "\n",
    "    for user, similarity in similar_users:\n",
    "        # Map the user index to its original userId\n",
    "        original_user_id = new_to_original_index[user]\n",
    "\n",
    "        for movie, sentiment in matrix.loc[user].items():\n",
    "            if not np.isnan(sentiment):\n",
    "\n",
    "                # Filter df_sent to get rows for the current user and movie\n",
    "                user_movie_df = df_sent[(df_sent['userId'] == original_user_id) & (df_sent['movieId'] == movie)]\n",
    "\n",
    "                # If there are no sentiments for this user-movie pair, continue\n",
    "                if user_movie_df.empty:\n",
    "                    continue\n",
    "\n",
    "                # Aggregate the scaled_sentiment_values. Here, I'm using mean, but you can adjust this\n",
    "                movie_sentiment = user_movie_df['scaled_sentiment_value'].mean()\n",
    "\n",
    "                sentiment_direction = get_dominant_sentiment([movie_sentiment])\n",
    "                sentiment_buckets[sentiment_direction].append((movie, movie_sentiment))\n",
    "\n",
    "    return sentiment_buckets[dominant_sentiment][\n",
    "           :k]  # returns movieId and sentiment value - scaled sentiment value from df_sent\n",
    "\n",
    "\n",
    "def generate_and_store_recommendations(df_rec, mat, new_to_original_index, df_sent, k=10):\n",
    "    for idx, row in df_rec.iterrows():\n",
    "        original_user_id = row['userId']\n",
    "\n",
    "        # Map the original userId to the new continuous index\n",
    "        target_user = original_to_new_index.get(original_user_id, None)\n",
    "\n",
    "        # If the user exists in the matrix\n",
    "        if target_user is not None:\n",
    "            # Get recommendations\n",
    "            recommendations = get_distributed_recommendations(target_user, mat, new_to_original_index, df_sent, k)\n",
    "\n",
    "            # Store recommendations in df_rec under the original userId\n",
    "            df_rec.at[idx, 'recommendations'] = recommendations\n",
    "\n",
    "\n",
    "# Define sentiment category function\n",
    "def determine_sentiment_category(sentiment_value):\n",
    "    if -1 <= sentiment_value < -0.5:\n",
    "        return 'Negative'\n",
    "    elif -0.5 <= sentiment_value <= 0.5:\n",
    "        return 'Neutral'\n",
    "    else:\n",
    "        return 'Positive'\n",
    "\n",
    "\n",
    "# Extract tags function\n",
    "def get_tags_for_movie(movieId, sentiment_value):\n",
    "    sentiment_category = determine_sentiment_category(sentiment_value)\n",
    "\n",
    "    # Filter rows with matching movieId and sentiment category\n",
    "    tags = df_comb_t[(df_comb_t['movieId'] == movieId) &\n",
    "                     (df_comb_t['scaled_sentiment_value_avg'].apply(\n",
    "                         determine_sentiment_category) == sentiment_category)]['tag'].tolist()\n",
    "\n",
    "    # Convert tags list to set and back to list to ensure distinct tags\n",
    "    distinct_tags = list(set(tags))\n",
    "\n",
    "    return (movieId, distinct_tags)"
   ],
   "metadata": {
    "collapsed": false
   },
   "id": "8f69bafe13d86a07"
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Training"
   ],
   "metadata": {
    "collapsed": false
   },
   "id": "dce0ff635c3b81ed"
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "initial_id",
   "metadata": {
    "collapsed": true,
    "ExecuteTime": {
     "end_time": "2024-09-04T12:17:41.392343Z",
     "start_time": "2024-09-04T10:23:31.276513Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "No recommendations for user 11585, adding empty recommendation.\n",
      "No recommendations for user 13494, adding empty recommendation.\n",
      "No recommendations for user 13920, adding empty recommendation.\n",
      "No recommendations for user 27860, adding empty recommendation.\n",
      "No recommendations for user 29096, adding empty recommendation.\n",
      "No recommendations for user 31935, adding empty recommendation.\n",
      "No recommendations for user 50687, adding empty recommendation.\n",
      "No recommendations for user 54248, adding empty recommendation.\n",
      "No recommendations for user 77374, adding empty recommendation.\n",
      "No recommendations for user 98756, adding empty recommendation.\n",
      "No recommendations for user 115495, adding empty recommendation.\n",
      "No recommendations for user 117259, adding empty recommendation.\n",
      "No recommendations for user 120748, adding empty recommendation.\n",
      "No recommendations for user 126491, adding empty recommendation.\n",
      "No recommendations for user 127969, adding empty recommendation.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/lib/function_base.py:518: RuntimeWarning: Mean of empty slice.\n",
      "  avg = a.mean(axis, **keepdims_kw)\n",
      "/Users/jiayi/anaconda3/lib/python3.11/site-packages/numpy/core/_methods.py:192: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average Precision: 0.2241121495327102\n",
      "Average Recall: 1.0\n",
      "Average Accuracy: nan\n",
      "Overall Precision: 0.811214953271028\n",
      "Overall Recall: 0.3638383873023033\n",
      "The overall average similarity across all users is: 0.9321706839264308\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001B[0;31m---------------------------------------------------------------------------\u001B[0m",
      "\u001B[0;31mKeyboardInterrupt\u001B[0m                         Traceback (most recent call last)",
      "Cell \u001B[0;32mIn[47], line 484\u001B[0m\n\u001B[1;32m    479\u001B[0m recommendations, user_tags, tags_movies \u001B[38;5;241m=\u001B[39m get_user_data(df_rec, tag_cv_filtered, target_user)\n\u001B[1;32m    481\u001B[0m user_features, user_wsd \u001B[38;5;241m=\u001B[39m get_user_feature_set_and_wsd(df_feat, df_wsd_filtered, user_tags, tag_clusters,\n\u001B[1;32m    482\u001B[0m                                                        target_user)\n\u001B[0;32m--> 484\u001B[0m similarity_df \u001B[38;5;241m=\u001B[39m calculate_similarity_for_recommendations(recommendations, tags_movies, df_feat, user_features,\n\u001B[1;32m    485\u001B[0m                                                          df_wsd, user_wsd)\n\u001B[1;32m    487\u001B[0m weights \u001B[38;5;241m=\u001B[39m {\n\u001B[1;32m    488\u001B[0m     \u001B[38;5;124m'\u001B[39m\u001B[38;5;124mwsd_similarity\u001B[39m\u001B[38;5;124m'\u001B[39m: \u001B[38;5;241m0.05\u001B[39m,\n\u001B[1;32m    489\u001B[0m     \u001B[38;5;124m'\u001B[39m\u001B[38;5;124mwsd_confidence\u001B[39m\u001B[38;5;124m'\u001B[39m: \u001B[38;5;241m0.05\u001B[39m,\n\u001B[0;32m   (...)\u001B[0m\n\u001B[1;32m    493\u001B[0m     \u001B[38;5;124m'\u001B[39m\u001B[38;5;124mcluster_match\u001B[39m\u001B[38;5;124m'\u001B[39m: \u001B[38;5;241m0.1\u001B[39m\n\u001B[1;32m    494\u001B[0m }\n\u001B[1;32m    496\u001B[0m \u001B[38;5;66;03m# Add a check for empty DataFrame\u001B[39;00m\n",
      "Cell \u001B[0;32mIn[47], line 404\u001B[0m, in \u001B[0;36mcalculate_similarity_for_recommendations\u001B[0;34m(recommendations, tags_movies, df_feat, user_features, df_wsd, user_wsd)\u001B[0m\n\u001B[1;32m    401\u001B[0m \u001B[38;5;28;01mfor\u001B[39;00m movie_tag \u001B[38;5;129;01min\u001B[39;00m movie_tags:\n\u001B[1;32m    403\u001B[0m     user_wsd_row \u001B[38;5;241m=\u001B[39m user_wsd[user_wsd[\u001B[38;5;124m'\u001B[39m\u001B[38;5;124mtag\u001B[39m\u001B[38;5;124m'\u001B[39m] \u001B[38;5;241m==\u001B[39m user_tag]\n\u001B[0;32m--> 404\u001B[0m     movie_wsd_row \u001B[38;5;241m=\u001B[39m df_wsd[(df_wsd[\u001B[38;5;124m'\u001B[39m\u001B[38;5;124mmovieId\u001B[39m\u001B[38;5;124m'\u001B[39m] \u001B[38;5;241m==\u001B[39m movieId) \u001B[38;5;241m&\u001B[39m (df_wsd[\u001B[38;5;124m'\u001B[39m\u001B[38;5;124mtag\u001B[39m\u001B[38;5;124m'\u001B[39m] \u001B[38;5;241m==\u001B[39m movie_tag)]\n\u001B[1;32m    406\u001B[0m     \u001B[38;5;66;03m# nested loop to calculate the wsd\u001B[39;00m\n\u001B[1;32m    407\u001B[0m     wsd_similarities \u001B[38;5;241m=\u001B[39m []\n",
      "File \u001B[0;32m~/anaconda3/lib/python3.11/site-packages/pandas/core/ops/common.py:72\u001B[0m, in \u001B[0;36m_unpack_zerodim_and_defer.<locals>.new_method\u001B[0;34m(self, other)\u001B[0m\n\u001B[1;32m     68\u001B[0m             \u001B[38;5;28;01mreturn\u001B[39;00m \u001B[38;5;28mNotImplemented\u001B[39m\n\u001B[1;32m     70\u001B[0m other \u001B[38;5;241m=\u001B[39m item_from_zerodim(other)\n\u001B[0;32m---> 72\u001B[0m \u001B[38;5;28;01mreturn\u001B[39;00m method(\u001B[38;5;28mself\u001B[39m, other)\n",
      "File \u001B[0;32m~/anaconda3/lib/python3.11/site-packages/pandas/core/arraylike.py:42\u001B[0m, in \u001B[0;36mOpsMixin.__eq__\u001B[0;34m(self, other)\u001B[0m\n\u001B[1;32m     40\u001B[0m \u001B[38;5;129m@unpack_zerodim_and_defer\u001B[39m(\u001B[38;5;124m\"\u001B[39m\u001B[38;5;124m__eq__\u001B[39m\u001B[38;5;124m\"\u001B[39m)\n\u001B[1;32m     41\u001B[0m \u001B[38;5;28;01mdef\u001B[39;00m \u001B[38;5;21m__eq__\u001B[39m(\u001B[38;5;28mself\u001B[39m, other):\n\u001B[0;32m---> 42\u001B[0m     \u001B[38;5;28;01mreturn\u001B[39;00m \u001B[38;5;28mself\u001B[39m\u001B[38;5;241m.\u001B[39m_cmp_method(other, operator\u001B[38;5;241m.\u001B[39meq)\n",
      "File \u001B[0;32m~/anaconda3/lib/python3.11/site-packages/pandas/core/series.py:6243\u001B[0m, in \u001B[0;36mSeries._cmp_method\u001B[0;34m(self, other, op)\u001B[0m\n\u001B[1;32m   6240\u001B[0m rvalues \u001B[38;5;241m=\u001B[39m extract_array(other, extract_numpy\u001B[38;5;241m=\u001B[39m\u001B[38;5;28;01mTrue\u001B[39;00m, extract_range\u001B[38;5;241m=\u001B[39m\u001B[38;5;28;01mTrue\u001B[39;00m)\n\u001B[1;32m   6242\u001B[0m \u001B[38;5;28;01mwith\u001B[39;00m np\u001B[38;5;241m.\u001B[39merrstate(\u001B[38;5;28mall\u001B[39m\u001B[38;5;241m=\u001B[39m\u001B[38;5;124m\"\u001B[39m\u001B[38;5;124mignore\u001B[39m\u001B[38;5;124m\"\u001B[39m):\n\u001B[0;32m-> 6243\u001B[0m     res_values \u001B[38;5;241m=\u001B[39m ops\u001B[38;5;241m.\u001B[39mcomparison_op(lvalues, rvalues, op)\n\u001B[1;32m   6245\u001B[0m \u001B[38;5;28;01mreturn\u001B[39;00m \u001B[38;5;28mself\u001B[39m\u001B[38;5;241m.\u001B[39m_construct_result(res_values, name\u001B[38;5;241m=\u001B[39mres_name)\n",
      "File \u001B[0;32m~/anaconda3/lib/python3.11/site-packages/pandas/core/ops/array_ops.py:287\u001B[0m, in \u001B[0;36mcomparison_op\u001B[0;34m(left, right, op)\u001B[0m\n\u001B[1;32m    284\u001B[0m     \u001B[38;5;28;01mreturn\u001B[39;00m invalid_comparison(lvalues, rvalues, op)\n\u001B[1;32m    286\u001B[0m \u001B[38;5;28;01melif\u001B[39;00m is_object_dtype(lvalues\u001B[38;5;241m.\u001B[39mdtype) \u001B[38;5;129;01mor\u001B[39;00m \u001B[38;5;28misinstance\u001B[39m(rvalues, \u001B[38;5;28mstr\u001B[39m):\n\u001B[0;32m--> 287\u001B[0m     res_values \u001B[38;5;241m=\u001B[39m comp_method_OBJECT_ARRAY(op, lvalues, rvalues)\n\u001B[1;32m    289\u001B[0m \u001B[38;5;28;01melse\u001B[39;00m:\n\u001B[1;32m    290\u001B[0m     res_values \u001B[38;5;241m=\u001B[39m _na_arithmetic_op(lvalues, rvalues, op, is_cmp\u001B[38;5;241m=\u001B[39m\u001B[38;5;28;01mTrue\u001B[39;00m)\n",
      "File \u001B[0;32m~/anaconda3/lib/python3.11/site-packages/pandas/core/ops/array_ops.py:75\u001B[0m, in \u001B[0;36mcomp_method_OBJECT_ARRAY\u001B[0;34m(op, x, y)\u001B[0m\n\u001B[1;32m     73\u001B[0m     result \u001B[38;5;241m=\u001B[39m libops\u001B[38;5;241m.\u001B[39mvec_compare(x\u001B[38;5;241m.\u001B[39mravel(), y\u001B[38;5;241m.\u001B[39mravel(), op)\n\u001B[1;32m     74\u001B[0m \u001B[38;5;28;01melse\u001B[39;00m:\n\u001B[0;32m---> 75\u001B[0m     result \u001B[38;5;241m=\u001B[39m libops\u001B[38;5;241m.\u001B[39mscalar_compare(x\u001B[38;5;241m.\u001B[39mravel(), y, op)\n\u001B[1;32m     76\u001B[0m \u001B[38;5;28;01mreturn\u001B[39;00m result\u001B[38;5;241m.\u001B[39mreshape(x\u001B[38;5;241m.\u001B[39mshape)\n",
      "\u001B[0;31mKeyboardInterrupt\u001B[0m: "
     ]
    }
   ],
   "source": [
    "# fold 1 \n",
    "# reading in fold 1\n",
    "for fold in range(1,5):\n",
    "    df_fold_train_1 = pd.read_csv(f\"../dataset/evaluation_folds/fold_{fold}_train.csv\")\n",
    "    tags_c = pd.read_csv('../dataset/final_clusters.csv')\n",
    "    tags = pd.read_csv(\"../dataset/tags_withglovevec.csv\")\n",
    "    df_s = pd.read_csv('../dataset/df_tag_sentiment.csv')\n",
    "    \n",
    "    # data conversions\n",
    "    tags_c['tag'] = tags_c['tag'].astype('str')\n",
    "    df_s['tag'] = df_s['tag'].astype('str')\n",
    "        \n",
    "    # combine df_comb and tags_c\n",
    "    df_comb = tags_c.merge(df_s, on='tag', how='left')\n",
    "    \n",
    "    # drop duplicates on (tag, movie) -> so the cluster average is not bias to duplicate tags per movie\n",
    "    df_comb.drop_duplicates(subset='tag', inplace=True)\n",
    "    # df_comb.drop(columns=['Unnamed: 0'], inplace=True)\n",
    "    # df_comb\n",
    "    avg_sentiment_by_cluster = df_comb.groupby('cluster')['scaled_sentiment_value'].mean().reset_index()\n",
    "    \n",
    "    df_comb = pd.merge(df_comb, avg_sentiment_by_cluster, on='cluster', suffixes=('', '_cluster_avg'))\n",
    "    df_comb = df_comb.drop(columns=['glove_vec'])\n",
    "    df_comb['scaled_sentiment_value_avg'] = df_comb['scaled_sentiment_value_cluster_avg']\n",
    "    df_mat = tags.merge(df_comb, on=['tag'], how='inner')\n",
    "    \n",
    "    df_sent = df_mat  # take copy for later on \n",
    "  \n",
    "    #### CF generate recommendations and store in a file here - \n",
    "    # df_fold_train_1\n",
    "    # tags_c\n",
    "    tags_c['cluster'] = tags_c['new_cluster']\n",
    "    tags_c.drop(columns='new_cluster', inplace=True)\n",
    "    \n",
    "    df_s = pd.read_csv('../dataset/df_tag_sentiment.csv')\n",
    "    \n",
    "    # data conversions\n",
    "    tags_c['tag'] = tags_c['tag'].astype('str')\n",
    "    df_s['tag'] = df_s['tag'].astype('str')\n",
    "        \n",
    "    # combine df_comb and tags_c\n",
    "    df_comb = tags_c.merge(df_s, on='tag', how='left')\n",
    "    \n",
    "    # drop duplicates on (tag, movie) -> so the cluster average is not bias to duplicate tags per movie\n",
    "    df_comb.drop_duplicates(subset='tag', inplace=True)\n",
    "    \n",
    "    avg_sentiment_by_cluster = df_comb.groupby('cluster')['scaled_sentiment_value'].mean().reset_index()\n",
    "    \n",
    "    df_comb = pd.merge(df_comb, avg_sentiment_by_cluster, on='cluster', suffixes=('', '_cluster_avg'))\n",
    "    \n",
    "    df_mat = df_fold_train_1.merge(df_comb, on=['tag'], how='inner')\n",
    "    df_sent = df_mat\n",
    "    # df_mat\n",
    "    df_mat['scaled_sentiment_value_avg'] = df_mat['scaled_sentiment_value_cluster_avg']\n",
    "    \n",
    "    # Convert the 'scaled_sentiment_value_avg' column to float\n",
    "    df_mat['scaled_sentiment_value_avg'] = df_mat['scaled_sentiment_value_avg'].astype('float')\n",
    "    \n",
    "    # Keep only relevant columns\n",
    "    df_mat = df_mat[[\"userId\", \"movieId\", \"scaled_sentiment_value_avg\"]]\n",
    "    \n",
    "    # Check for duplicates and print them\n",
    "    duplicates = df_mat[df_mat.duplicated(subset=['userId', 'movieId'], keep=False)]\n",
    "    \n",
    "    # Group by 'userId' and 'movieId' to get the average 'scaled_sentiment_value_avg'\n",
    "    df_mat = df_mat.groupby(['userId', 'movieId'])['scaled_sentiment_value_avg'].mean().reset_index()\n",
    "    \n",
    "    # Create the pivot table\n",
    "    mat = pd.pivot_table(df_mat, values='scaled_sentiment_value_avg', index=['userId'], columns=['movieId'])\n",
    "    \n",
    "    # Assuming 'mat' is your matrix\n",
    "    # Resetting the index will add a column 'userId' with the original values\n",
    "    mat = mat.reset_index()\n",
    "    \n",
    "    # creating a dictionary with original userId and new continuous index\n",
    "    original_to_new_index = {old_id: new_id for new_id, old_id in enumerate(mat['userId'])}\n",
    "    \n",
    "    # assign new continous index to the userId column \n",
    "    mat['userId'] = mat['userId'].map(original_to_new_index)\n",
    "    \n",
    "    # Now, set 'userId' as the index again\n",
    "    mat.set_index('userId', inplace=True)\n",
    "    \n",
    "    user_indices = mat.index[mat.apply(lambda x: (x.count() == 1) and (x.dropna().iloc[0] > 0.5), axis=1)]\n",
    "    \n",
    "    new_to_original_index = {v: k for k, v in\n",
    "                             original_to_new_index.items()}  # this exists outside the function to ensure the new_to_original_index exists for mapping\n",
    "    \n",
    "    original_to_new_index = {v: k for k, v in new_to_original_index.items()}\n",
    "    df_comb['scaled_sentiment_value_avg'] = df_comb['scaled_sentiment_value_cluster_avg']\n",
    "    # creation of df_rec dataframe \n",
    "    \n",
    "    df_rec = df_sent[[\"userId\"]].drop_duplicates().reset_index(drop=True)\n",
    "    \n",
    "    # Initialize recommendations column in df_rec\n",
    "    df_rec['recommendations'] = None\n",
    "    \n",
    "    # Generate and store recommendations\n",
    "    generate_and_store_recommendations(df_rec, mat, new_to_original_index, df_sent)\n",
    "    df_rec['recommendations'] = df_rec['recommendations'].apply(lambda x: [(int(a), round(b, 4)) for a, b in x])\n",
    "    \n",
    "    df_comb_t = pd.merge(tags, df_comb, on='tag', how='left')\n",
    "    df_comb_t = df_comb_t[[\"movieId\", \"tag\", \"scaled_sentiment_value\", \"scaled_sentiment_value_avg\"]]\n",
    "    \n",
    "    # getting initial tags recommendation \n",
    "    # Apply function on recommendations column\n",
    "    df_rec['tags_movies'] = df_rec['recommendations'].apply(\n",
    "        lambda recs: [get_tags_for_movie(movieId, sentiment_value) for movieId, sentiment_value in recs])\n",
    "    df_rec.to_json(f\"../dataset/evaluation/cf_train_{fold}_recs.json\")\n",
    "    \n",
    "    \n",
    "    df_feat = pd.read_json(\"../dataset/df_feat.json\")\n",
    "    tag_cv = pd.read_json('../dataset/tag_csv.json')\n",
    "    tag_clusters = pd.read_csv(\"../dataset/tag_clusters.csv\")\n",
    "    \n",
    "    df_wsd = pd.read_csv(\"../dataset/df_wsd.csv\")\n",
    "    # df_wsd  # can directly filter on this\n",
    "    # df_feat  # need to merge on tag_clusters\n",
    "    df_feat_cl = pd.merge(df_feat, tag_clusters, on=['tag'], how='left')\n",
    "\n",
    "    \n",
    "    def get_user_data(df_rec, tag_cv, target_userId):\n",
    "        \"\"\"\n",
    "        Retrieve movie recommendations, user tags, and tags_movies for a target userId.\n",
    "    \n",
    "        Parameters:\n",
    "            df_rec (DataFrame): The DataFrame containing userId, recommendations, and tags_movies columns.\n",
    "            tag_cv (DataFrame): The DataFrame containing userId, movieId, and tags.\n",
    "            target_userId (int): The userId for whom the data is to be fetched.\n",
    "    \n",
    "        Returns:\n",
    "            tuple: A tuple containing three lists:\n",
    "                   - The first list contains recommendations (movieId, sentiment) for the target userId.\n",
    "                   - The second list contains tags applied by the target userId in the format (movieId, tag).\n",
    "                   - The third list contains tags_movies entries for the target userId in the format [(movieId, [tags...]), ...].\n",
    "        \"\"\"\n",
    "        # Fetch movie recommendations\n",
    "        recommendations = get_user_recommendations(df_rec, target_userId)\n",
    "    \n",
    "        # Fetch tags applied by the user\n",
    "        user_tags = get_user_tags(tag_cv, target_userId)\n",
    "    \n",
    "        # Fetch tags_movies data for the user\n",
    "        tags_movies = get_tags_movies_for_user(df_rec, target_userId)\n",
    "    \n",
    "        return recommendations, user_tags, tags_movies\n",
    "    \n",
    "    \n",
    "    def get_user_recommendations(df_rec, target_userId):\n",
    "        filtered_df = df_rec[df_rec['userId'] == target_userId]\n",
    "        if len(filtered_df) == 0:\n",
    "            return []\n",
    "        recommendations = filtered_df['recommendations'].iloc[0]\n",
    "        return recommendations\n",
    "    \n",
    "    \n",
    "    def get_user_tags(tag_cv, target_userId):\n",
    "        filtered_df = tag_cv[tag_cv['userId'] == target_userId]\n",
    "        if len(filtered_df) == 0:\n",
    "            return []\n",
    "        user_tags = filtered_df[['movieId', 'tag']].values.tolist()\n",
    "        return user_tags\n",
    "    \n",
    "    \n",
    "    def get_tags_movies_for_user(df_rec, target_userId):\n",
    "        filtered_df = df_rec[df_rec['userId'] == target_userId]\n",
    "        if len(filtered_df) == 0:\n",
    "            return []\n",
    "        tags_movies = filtered_df['tags_movies'].iloc[0]\n",
    "        return tags_movies\n",
    "    \n",
    "    \n",
    "    def get_user_feature_set_and_wsd(df_feat, df_wsd, user_tags, tag_clusters, target_userId):\n",
    "        \"\"\"\n",
    "        Fetch additional features, WSD, and TF/TF-IDF info for a list of user tags.\n",
    "        \n",
    "        Parameters:\n",
    "            df_feat (DataFrame): The DataFrame containing features for each tag.\n",
    "            df_wsd (DataFrame): The DataFrame containing WSD information.\n",
    "            user_tags (list): The list containing tags applied by the user in the format (movieId, tag).\n",
    "            target_userId (int): The userId for whom the WSD data is to be fetched.\n",
    "            \n",
    "        Returns:\n",
    "            tuple: A tuple containing three DataFrames:\n",
    "                   - The first DataFrame contains additional features for each user tag.\n",
    "                   - The second DataFrame contains WSD information for the user.\n",
    "                   - The third DataFrame contains TF and TF-IDF information for the user tags.\n",
    "        \"\"\"\n",
    "        # Filtering df_feat to only include rows where the tag is in user_tags\n",
    "        filtered_df_feat = df_feat[df_feat['tag'].isin([tag for _, tag in user_tags])]\n",
    "\n",
    "        # Filtering df_wsd to include only rows for the target_userId\n",
    "        filtered_df_wsd = df_wsd[df_wsd['userId'] == target_userId]\n",
    "    \n",
    "        # joining the cluster to the tags\n",
    "    \n",
    "        # Create a set of (movieId, tag) tuples for easier filtering\n",
    "        user_tags_set = set(tuple(x) for x in user_tags)\n",
    "    \n",
    "        return filtered_df_feat, filtered_df_wsd\n",
    "    \n",
    "    \n",
    "    from sklearn.metrics.pairwise import cosine_similarity\n",
    "    import numpy as np\n",
    "    import pandas as pd\n",
    "\n",
    "\n",
    "    def calculate_similarity_for_recommendations(recommendations, tags_movies, df_feat, user_features, df_wsd, user_wsd):\n",
    "        similarity_list = []\n",
    "\n",
    "        # Create a dictionary to easily fetch movie tags\n",
    "        tags_movies_dict = dict(tags_movies)\n",
    "    \n",
    "        for movieId, _ in recommendations:\n",
    "            movie_tags = tags_movies_dict.get(movieId, [])  # Fetch the tags from the tags_movies dictionary\n",
    "            if not movie_tags:\n",
    "                continue\n",
    "    \n",
    "            for user_tag in user_features['tag'].unique():\n",
    "                for movie_tag in movie_tags:\n",
    "    \n",
    "                    user_wsd_row = user_wsd[user_wsd['tag'] == user_tag]\n",
    "                    movie_wsd_row = df_wsd[(df_wsd['movieId'] == movieId) & (df_wsd['tag'] == movie_tag)]\n",
    "    \n",
    "                    # nested loop to calculate the wsd\n",
    "                    wsd_similarities = []\n",
    "                    for user_sense in user_wsd_row['disambiguated_sense'].values:\n",
    "                        for movie_sense in movie_wsd_row['disambiguated_sense'].values:\n",
    "                            wsd_similarities.append(user_sense == movie_sense)\n",
    "    \n",
    "                    if wsd_similarities:\n",
    "                        wsd_similarity = np.mean(wsd_similarities)\n",
    "                    else:\n",
    "                        wsd_similarity = 0\n",
    "    \n",
    "                    wsd_confidence = movie_wsd_row['confidence'].mean() if not movie_wsd_row.empty else 0\n",
    "    \n",
    "                    user_feat_row = user_features[user_features['tag'] == user_tag]\n",
    "                    movie_feat_row = df_feat[df_feat['tag'] == movie_tag]\n",
    "    \n",
    "                    pos_match = (user_feat_row['POS'].values == movie_feat_row['POS'].values).mean() if user_feat_row['POS'].values.size and movie_feat_row['POS'].values.size else 0\n",
    "                    ner_match = (user_feat_row['ner_label'].values == movie_feat_row['ner_label'].values).mean() if user_feat_row['ner_label'].values.size and movie_feat_row['ner_label'].values.size else 0\n",
    "                    sentiment_match = (user_feat_row['sentiment_label'].values == movie_feat_row['sentiment_label'].values).mean() if user_feat_row['sentiment_label'].values.size and movie_feat_row['sentiment_label'].values.size else 0\n",
    "    \n",
    "                    # Check for cluster match\n",
    "                    cluster_match = int(user_feat_row['cluster'].values[0] == movie_feat_row['cluster'].values[0]) if user_feat_row['cluster'].values.size and movie_feat_row['cluster'].values.size else 0\n",
    "    \n",
    "                    similarity_list.append({\n",
    "                        'movieId': movieId,\n",
    "                        'user_tag': user_tag,\n",
    "                        'movie_tag': movie_tag,\n",
    "                        'wsd_similarity': wsd_similarity,\n",
    "                        'wsd_confidence': wsd_confidence,\n",
    "                        'pos_match': pos_match,\n",
    "                        'ner_match': ner_match,\n",
    "                        'sentiment_match': sentiment_match,\n",
    "                        'cluster_match': cluster_match  # Adding the cluster match to the dataframe\n",
    "                    })\n",
    "    \n",
    "        similarity_df = pd.DataFrame(similarity_list)\n",
    "        return similarity_df\n",
    "    \n",
    "    \n",
    "    def calculate_total_similarity(row, weights):\n",
    "        return sum(row[metric] * weight for metric, weight in weights.items())\n",
    "    \n",
    "            \n",
    "    precision_scores = []\n",
    "    recall_scores = []\n",
    "    accuracy_scores = []\n",
    "    \n",
    "    # df_feat\n",
    "    \n",
    "    test_data = pd.read_csv(f'../dataset/evaluation_folds/fold_{fold}_test.csv')\n",
    "    train_data = df_fold_train_1\n",
    "    \n",
    "    df_fold_1_results = pd.DataFrame(columns=['userId', 'recommended_tags'])\n",
    "    \n",
    "    ############ CB MODEL ############################\n",
    "    \n",
    "    merged_tag_cv = pd.merge(tag_cv, test_data, on=['userId', 'movieId', 'tag'], how='outer', indicator=True)\n",
    "    \n",
    "    # Now, filter out the rows that come only from the 'test_data' using the indicator column.\n",
    "    tag_cv_filtered = merged_tag_cv[merged_tag_cv['_merge'] != 'right_only'].drop(columns=['_merge'])\n",
    "    \n",
    "    # Repeat the process for df_wsd.\n",
    "    merged_df_wsd = pd.merge(df_wsd, test_data, on=['userId', 'movieId', 'tag'], how='outer', indicator=True)\n",
    "    df_wsd_filtered = merged_df_wsd[merged_df_wsd['_merge'] != 'right_only'].drop(columns=['_merge'])\n",
    "    \n",
    "    for userId in train_data['userId'].unique():\n",
    "        try:\n",
    "            target_user = userId\n",
    "    \n",
    "            recommendations, user_tags, tags_movies = get_user_data(df_rec, tag_cv_filtered, target_user)\n",
    "    \n",
    "            user_features, user_wsd = get_user_feature_set_and_wsd(df_feat, df_wsd_filtered, user_tags, tag_clusters,\n",
    "                                                                   target_user)\n",
    "    \n",
    "            similarity_df = calculate_similarity_for_recommendations(recommendations, tags_movies, df_feat, user_features,\n",
    "                                                                     df_wsd, user_wsd)\n",
    "    \n",
    "            weights = {\n",
    "                'wsd_similarity': 0.05,\n",
    "                'wsd_confidence': 0.05,\n",
    "                'pos_match': 0.3,\n",
    "                'ner_match': 0.3,\n",
    "                'sentiment_match': 0.2,\n",
    "                'cluster_match': 0.1\n",
    "            }\n",
    "    \n",
    "            # Add a check for empty DataFrame\n",
    "            if similarity_df.empty:\n",
    "                print(f\"No recommendations for user {userId}, adding empty recommendation.\")\n",
    "                recommended_tags = []\n",
    "            else:\n",
    "                # Assuming similarity_df is your DataFrame\n",
    "                similarity_df['total_similarity'] = similarity_df.apply(\n",
    "                    lambda row: calculate_total_similarity(row, weights), axis=1)\n",
    "    \n",
    "                # Sort the DataFrame by 'total_similarity' in descending order\n",
    "                sorted_similarity_df = similarity_df.sort_values(by='total_similarity', ascending=False)\n",
    "    \n",
    "                # Remove rows where 'movie_tag' is also present in 'user_tag' column\n",
    "                filtered_similarity_df = sorted_similarity_df[~sorted_similarity_df['movie_tag'].isin(user_tags)]\n",
    "    \n",
    "                # Drop duplicates based on the 'movie_tag' column to get distinct tags\n",
    "                distinct_tags_df = filtered_similarity_df.drop_duplicates(subset=['movie_tag'])\n",
    "    \n",
    "                # Get the top 10 distinct tags along with their movieId\n",
    "                top_10_distinct_tags_with_movieId = distinct_tags_df[\n",
    "                    ['movieId', 'movie_tag', 'ner_match', 'pos_match', 'cluster_match', 'total_similarity']].head(10)\n",
    "                # Calculate the average similarity of the top 10 tags\n",
    "                average_similarity = top_10_distinct_tags_with_movieId[\n",
    "                    'total_similarity'].mean() if not top_10_distinct_tags_with_movieId.empty else 0\n",
    "    \n",
    "                recommended_tags = top_10_distinct_tags_with_movieId['movie_tag'].tolist()\n",
    "    \n",
    "            # Now, we can add the results to df_fold_1_results DataFrame, regardless of whether recommendations are empty or not.\n",
    "            temp_df = pd.DataFrame({\n",
    "                'userId': [userId],  # Make sure to pass a list even if it's a single userId\n",
    "                'recommended_tags': [recommended_tags],\n",
    "                # The brackets ensure it's treated as a single list entry in the DataFrame\n",
    "                'average_similarity': [average_similarity]  # Add the average similarity for the top 10 tags\n",
    "            })\n",
    "    \n",
    "            # Concatenate the temporary DataFrame with the main results DataFrame\n",
    "            df_fold_1_results = pd.concat([df_fold_1_results, temp_df], ignore_index=True)\n",
    "    \n",
    "        except Exception as e:\n",
    "            print(f\"An error occurred for user {userId}: {e}\")\n",
    "    \n",
    "    ### RESULTs\n",
    "    # df_fold_1_results.to_csv(f'../dataset/evaluation/df_fold_{fold}_results.csv',index=False)\n",
    "    df_fold_1_results.to_json(f'../dataset/evaluation/df_fold_{fold}_results.json')\n",
    "    print(f'Finished Recommendation fold {fold}')\n"
   ]
  },
  {
   "cell_type": "code",
   "outputs": [],
   "source": [
    "# test_data_0 = pd.read_csv('../dataset/evaluation_folds/fold_4_test.csv')\n",
    "# # df_fold_0_results = pd.read_csv('../dataset/evaluation/df_fold_0_results.csv')\n",
    "# \n",
    "# df_fold_0_results = pd.read_json(\"../dataset/evaluation/df_fold_4_results.json\")\n",
    "# # test_data_0 = pd.read_json('../dataset/evaluation/fold_0_test.json')\n"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2024-09-04T12:19:28.331577Z",
     "start_time": "2024-09-04T12:19:28.210104Z"
    }
   },
   "id": "94a14530a802c91b",
   "execution_count": 56
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Testing"
   ],
   "metadata": {
    "collapsed": false
   },
   "id": "bda476723e337144"
  },
  {
   "cell_type": "code",
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "            Fold  Precision    Recall  F1 Score  Average Similarity\n",
      "0              0   0.808194  0.359797  0.497924            0.933194\n",
      "1              1   0.811215  0.363838  0.502362            0.932171\n",
      "2              2   0.813084  0.360019  0.499062            0.932588\n",
      "3              3   0.799625  0.355138  0.491837            0.933250\n",
      "4              4   0.831776  0.381844  0.523407            0.935163\n",
      "Average  Average   0.812779  0.364127  0.502918            0.933273\n"
     ]
    }
   ],
   "source": [
    "import pandas as pd\n",
    "from collections import defaultdict\n",
    "\n",
    "# Define a function to calculate precision, recall, F1 score, and average similarity\n",
    "def calculate_metrics(test_data_path, results_data_path):\n",
    "    # Read the test data and results data\n",
    "    test_data = pd.read_csv(test_data_path)\n",
    "    df_results = pd.read_json(results_data_path)\n",
    "\n",
    "    # Process the test data into a mapping of userId to a list of actual tags\n",
    "    test_tags_dict = defaultdict(set)\n",
    "    for _, row in test_data.iterrows():\n",
    "        test_tags_dict[row['userId']].add(row['tag'])\n",
    "\n",
    "    # Calculate precision\n",
    "    individual_precisions = []\n",
    "    for _, row in df_results.iterrows():\n",
    "        user_id = row['userId']\n",
    "        recommended_tags = set(row['recommended_tags'])\n",
    "        actual_tags = test_tags_dict.get(user_id, set())\n",
    "        if recommended_tags.intersection(actual_tags):\n",
    "            individual_precisions.append(1)\n",
    "        else:\n",
    "            individual_precisions.append(0)\n",
    "\n",
    "    overall_precision = sum(individual_precisions) / len(individual_precisions)\n",
    "\n",
    "    # Calculate recall\n",
    "    individual_recalls = {}\n",
    "    for _, row in df_results.iterrows():\n",
    "        user_id = row['userId']\n",
    "        recommended_tags = set(row['recommended_tags'])\n",
    "        actual_tags = test_tags_dict.get(user_id, set())\n",
    "        if actual_tags:\n",
    "            individual_recall = len(recommended_tags.intersection(actual_tags)) / len(actual_tags)\n",
    "            individual_recalls[user_id] = individual_recall\n",
    "        else:\n",
    "            individual_recalls[user_id] = 0\n",
    "\n",
    "    overall_recall = sum(individual_recalls.values()) / len(individual_recalls)\n",
    "\n",
    "    overall_f1 = 0.0\n",
    "    # Calculate F1 score\n",
    "    if overall_precision + overall_recall > 0:\n",
    "        overall_f1 = 2 * (overall_precision * overall_recall) / (overall_precision + overall_recall)\n",
    "\n",
    "    # OTS\n",
    "    if 'average_similarity' in df_results.columns:\n",
    "        overall_average_similarity = df_results['average_similarity'].mean()\n",
    "    else:\n",
    "        print(\"The 'average_similarity' column does not exist in the results DataFrame.\")\n",
    "        overall_average_similarity = None\n",
    "\n",
    "    return overall_precision, overall_recall, overall_f1, overall_average_similarity\n",
    "\n",
    "\n",
    "# Initialize a list to store all metrics for each fold\n",
    "metrics_data = []\n",
    "\n",
    "# Loop through fold_0 to fold_4\n",
    "for fold_num in range(5):\n",
    "    test_data_path = f'../dataset/evaluation_folds/fold_{fold_num}_test.csv'\n",
    "    results_data_path = f'../dataset/evaluation/df_fold_{fold_num}_results.json'\n",
    "\n",
    "    # Calculate precision, recall, F1 score, and average similarity\n",
    "    precision, recall, f1_score, average_similarity = calculate_metrics(test_data_path, results_data_path)\n",
    "\n",
    "    # Append results to the metrics data list\n",
    "    metrics_data.append({\n",
    "        'Fold': fold_num,\n",
    "        'Precision': precision,\n",
    "        'Recall': recall,\n",
    "        'F1 Score': f1_score,\n",
    "        'Average Similarity': average_similarity\n",
    "    })\n",
    "\n",
    "# Create a DataFrame from the collected metrics data\n",
    "df_metrics = pd.DataFrame(metrics_data)\n",
    "\n",
    "# Add a row for the overall average metrics across all folds\n",
    "df_metrics.loc['Average'] = {\n",
    "    'Fold': 'Average',\n",
    "    'Precision': df_metrics['Precision'].mean(),\n",
    "    'Recall': df_metrics['Recall'].mean(),\n",
    "    'F1 Score': df_metrics['F1 Score'].mean(),\n",
    "    'Average Similarity': df_metrics['Average Similarity'].mean() if df_metrics['Average Similarity'].notna().any() else None\n",
    "}\n",
    "\n",
    "# Output the table\n",
    "print(df_metrics)"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2024-09-20T23:07:55.863941Z",
     "start_time": "2024-09-20T23:07:54.923235Z"
    }
   },
   "id": "c2ac27ad531e0ea0",
   "execution_count": 2
  },
  {
   "cell_type": "markdown",
   "source": [
    "# GRAPHING ALL RESULTS "
   ],
   "metadata": {
    "collapsed": false
   },
   "id": "bb9615a0a0e5e245"
  },
  {
   "cell_type": "code",
   "outputs": [
    {
     "data": {
      "text/plain": "<Figure size 2000x800 with 2 Axes>",
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAB8YAAAMWCAYAAACDduxsAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/H5lhTAAAACXBIWXMAAA9hAAAPYQGoP6dpAAC3rElEQVR4nOzdd5gV5d0//vfu0hYBpUnsPQgrKALGglFRUbFEfTRqjPUxqBRjTCxYsQu2r4qIjVhC7Fhi15CYaDR2RaPE9tiIkSgQlb67vz/4sXGzoIDI2XN8va6Lyz1z7pn5nHPfLM68Z+4pq62trQ0AAAAAAAAAlKjyQhcAAAAAAAAAAN8mwTgAAAAAAAAAJU0wDgAAAAAAAEBJE4wDAAAAAAAAUNIE4wAAAAAAAACUNME4AAAAAAAAACVNMA4AAAAAAABASROMAwAAAAAAAFDSBOMAQKNUW1v7nd4/AAAA8O0r9PF/ofcP8F0iGAfgGznggAPSuXPnen822GCDbL311jn99NMzbdq0ZVLHCSeckL59+35r7ZeF888/P5tsskk22mij3HXXXQ3e/+CDDxb4XW+++eY58sgj8/zzz9dr/9e//jWdO3fOX//614Xu46mnnsoOO+yQDTbYIIcddti3/REX2ahRo3Lttdd+ZZv5Y2/fffddaJtf/OIX6dy5c0444YTF2v9zzz2XAQMG1L2e/92PGzdusbazqL7t7QMAAIXhmLnwDjjggBxwwAELfX9Bx9rz+2mLLbbI4MGD88477yzDiv/jy7UvzeNG5x/+w/kHgO+WJoUuAIDi17Vr15x22ml1r+fMmZNXX301F110UV577bXcdNNNKSsr+1ZrGDhwYA488MBvrf237e9//3uuueaa/PjHP86PfvSjrL322gtte+SRR2brrbdOksyaNSsfffRRbrzxxuy///657LLLst122yVJqqqqcsstt2Tddddd6D4OO+yw1NTU5Kqrrkr79u2/9c+5qC655JIMHjz4a9uVl5fnxRdfzEcffZTvfe979d6bPn16/vCHPyzR/m+77ba89dZbS7QuAADAlzlmLg5fPtZOkhkzZuTVV1/N6NGjc+ihh+bBBx9M8+bNC1fgUuL8Q33OPwB8twjGAfjGWrVqlY022qjest69e+eLL77IpZdempdeeqnB+0vb6quv/q22/7ZNnTo1SbLzzjunV69eX9l29dVXb/B97rTTTvnpT3+ak046KZtuumlatWrVoF8WtI+pU6emd+/e2XzzzZfWR1mmunbtmjfffDMPPvhgDj744Hrv/eEPf0hlZWXatGlTmOIAAADimLlYLOhYe7PNNstyyy2XYcOG5amnnspWW21VmOKWIucflozzDwClwVTqAHxrNthggyTJpEmTksybeupXv/pVjjrqqGy00UY55JBDksy76njEiBHZaqutssEGG2TXXXfN/fffX29btbW1ue6667LTTjule/fu2X777XPttdfWPYfpv6d5e+WVV3LQQQelZ8+e6dGjRw4++OC8+OKLde//d/vq6uqMHTs2u+66a7p3756tt946F1xwQWbNmlVvnYMPPjh33HFH3fRfP/rRj/KnP/3pa7+L+++/P3vuuWd69OiRLbbYIqeeemrdlHmXXXZZ3dRoBx100BJNV9esWbMMGTIkU6dOzQMPPJCk/lRmC9pH586d8+GHH+auu+6qN+XZ3//+9xx++OHZeOONs/HGG2fQoEF5//336/Y1f7s333xzttlmm2y88cZ54oknkiTPPvtsfvrTn2bDDTfMJptskuOPPz6ffvpp3brjxo1L165d89JLL2WfffZJt27dss0229Sbtqxz585JkpEjR9b9vDAtW7bMVlttlQcffLDBe/fff3922GGHNGlS/zrA+Veob7/99tlggw2yww475MYbb6x7/4QTTsidd96ZDz/8sMH0YpMnT85RRx2VHj16ZJNNNskpp5ySL774ou79RRlHSfLwww9nt912S/fu3bPHHnvk9ddfb1D/9ddfnx133DHdunXLlltumWHDhuXzzz//yu8DAAAoHo6Z/7Ptq666Krvssku6d++ejTbaKPvuu2+eeuqpujaXXXZZtt9++/zxj3/MrrvuWncs99/TgE+aNCmDBw9Oz549s8UWW+TXv/7113XD11pQ2Dlp0qQcc8wx2WSTTbLhhhvmoIMOyt/+9rd6bT7//POceeaZ2XLLLbPRRhvlf/7nf/LHP/6x7v2ZM2fmwgsvTL9+/bLBBhtk4403ziGHHJLXXnvtG9Xr/MM8zj/M4/wDQH2CcQC+NfOfwbXaaqvVLXvggQey3HLL5Yorrshhhx2W2traDBo0KDfffHMOOeSQXHHFFenRo0d+8Ytf1DvAHTFiREaMGJG+fftm9OjR2WuvvXLBBRfkqquuarDfzz//PIcddljatm2byy67LBdffHFmzJiR//3f/81nn322wFpPPfXUnHvuudluu+1yxRVXZP/9989vfvObDBw4sO5EQjLv5MG1116bo446KpdffnkqKioyZMiQr3wu3KhRo3LMMcdko402yqWXXppBgwbloYceygEHHJCZM2dm7733zqmnnlpXx8iRIxfre55vs802S3l5eYNnfSVpsI+LL744t9xySzp27Jitttoqt9xyS6qqqvLOO+9k3333zSeffJLhw4fn7LPPzvvvv5/99tsvn3zySb1tjhw5Mscff3xOPfXU9OjRI88880wOPvjgtGjRIv/v//2/nHjiiXn66adz4IEHZubMmXXr1dTU5Oijj07//v1z1VVXZeONN86IESPy5z//OUlyyy23JEn22muvup+/Sv/+/eumM5vv888/z5/+9KfssssuDdoPGzYsl156aXbbbbeMHj06O+64Y84555xcfvnlSeZNGbjVVlulY8eOueWWW+pNpXfJJZdkpZVWyqhRo3LQQQfl1ltvrddfizKOxo8fn6OOOiqdO3fO5Zdfnp122inHHntsvRrvvffenH/++dl///1z7bXXZtCgQbn77rtz5plnfu33AQAAFAfHzPNccMEFGTVqVPbZZ59cc801OfPMMzN16tT8/Oc/z4wZM+raTZ48OWeccUYOPPDAXHXVVVl11VVz/PHH101DPX369Pz0pz/N3//+95x55pk55ZRTctttt+WFF15YpP6oqanJ3Llz6/58/vnneeKJJ3LhhRdmlVVWqbvz+dNPP82+++6bV199NaecckouvPDC1NTUZP/996+rpbq6Ooceemh+97vf5fDDD8+oUaOy9tprZ9CgQXn22WeTJMcdd1zuuOOODBgwIGPGjMnQoUPzxhtv5Je//GW973RxOP/g/IPzDwBfzVTqAHxjtbW1mTt3bt3radOm5emnn647YJ9/FXySNG3aNKeffnqaNWuWJHniiSfy5z//ORdffHH69++fJNlyyy0zY8aMXHDBBdlll10yffr03HDDDfnpT39a9z/wm2++eSZPnpxnnnkmhx9+eL163nzzzUyZMiUHHnhgNt544yTJ2muvnVtuuSVffPFFWrdu3aD97bffnl/+8pcZMGBAkmSLLbbIiiuumOOOOy5/+tOf6qZL++yzzzJu3Li6aeVatmyZn/70p3nqqaeyww47NPhupk2bliuuuCI//vGP6w4Mk+T73/9+9t9//9xxxx3Zf//9657Dte6666Zr166L2wVJkiZNmqRt27aZPHlyg/e+973v1dvHhhtumGTeld7t2rWrm/LstNNOS2VlZa677rq0atUqybwD3u222y7XXHNNjj/++Lpt/uQnP8mOO+5Y9/rCCy/MWmutlSuvvDIVFRVJkg033DA777xz3edM5o2XgQMHZu+9906S9OzZM4888kj++Mc/1l1JP7/mRZlOcOutt05lZWW96cweeeSRtG/fPj179qzX9p133smtt96aY445pq6v+/Tpk7Kyslx55ZX5yU9+ktVXXz3t2rVLs2bN6vY/ffr0JMkOO+yQoUOH1n0vTzzxRN1dDIs6ji6//PJ07949559/fpJ5433+9zff008/nVVXXTX7779/ysvLs8kmm6Rly5ZfeTIJAABonBwzL/yYOUk+/vjj/OIXv6i7yzhJmjdvniFDhmTixIl1x2UzZszI2Wefnc022yxJsuaaa2abbbbJY489lnXWWSd33nlnJk2alHvvvbfu+HfDDTfM9ttvv0j9dNJJJ+Wkk06qt6xly5bZYostcvzxx2e55ZZLMu/u2qlTp+amm27KKquskiT54Q9/mP79++eSSy7JpZdemj/96U956aWXcvnll9c9h3vTTTfN+++/n6eeeirdu3fPF198kZNPPrmuXzfZZJN8/vnnOe+88/Kvf/0rHTt2XKS653P+wfkH5x8Avp47xgH4xp555plUVVXV/dl8881zzDHHZIMNNsiFF16YsrKyurZrr7123QF+kjz55JMpKyvLVlttVe/K7L59+2by5Ml544038uKLL2bu3Lnp169fvf2efPLJueaaaxrUs95666Vdu3Y54ogjcuqpp+aRRx5Jhw4dcuyxx+Z73/teg/ZPP/10knnPvvqynXfeORUVFXVTfCVJu3bt6j1rbf72vnwV+5e9+OKLmT17doMrh3v16pVVVlmlbt9LS21tbb3ve3E99dRT2WSTTdKiRYu6vmjVqlV69eqVv/zlL/XadunSpe7nGTNm5KWXXspWW21Vd9Jn7ty5WW211bLOOuvUTXU2X48ePep+nn9wPP/gb3G1aNEiffv2rTed2X333ZeddtqpwXfx1FNPpba2Nn379m0w3mbNmpXnnnvuK/f1389fW3XVVfPvf/87yaKNo5kzZ+bVV1/NNttsU6/NTjvtVO/1pptumnfeeSd77rlnRo4cmQkTJmTXXXetd6IIAAAoDo6ZF37MnMwL6Q466KB8+umnefbZZ3PHHXfknnvuSZLMnj27Xtsvh5fztz3/WPLZZ5/N6quvXhfKJslKK620yM9vHzx4cG6//fbcdtttOe6449KsWbPsvPPO+X//7//Vu6v/ySefTJcuXdKpU6e6/igvL88Pf/jDuuPm5557Lk2bNq03VXl5eXluvvnmDB48OM2aNcu1116b/v3755///Geeeuqp3HzzzfnDH/6wwM+9KJx/cP7B+QeAr+eOcQC+saqqqpx++ulJkrKysjRv3jwrrbRS3RW/Xzb/Cuv5pk6dmtra2rqr1P/bxx9/XHeVart27RapnuWWWy5jx47NFVdckQceeCC33HJLWrRokR/96Ec5+eST651kSFK3/f++Gnv+FdBfnkqusrKyXpv5Bz41NTULrGX+tjt06NDgvQ4dOix0mrolMWPGjEybNm2BJzIW1dSpU3P//fc3eF5d0vD7b9myZd3P//73v1NTU5Orr746V199dYN1mzdvXu91ixYt6r0uLy9f4qniknkHdoMHD85HH32U5s2b58knn8zRRx/doN3UqVOTNDx4nO+f//znV+7nv/v/y3UvyjiaNm1aamtr07Zt23ptVlxxxXqv+/fvn5qamvz2t7/NqFGjctlll2WVVVbJr371q7q7CQAAgOLgmHnhx8xJMmHChJx++umZMGFCKisrs+6662bllVdOkgbHiV/efnl5eb0206ZNa3CsNb/uf/3rXwvd/3yrrLJKunXrliTp3r172rZtm6FDh6aioqKu/5J5ffLuu++mqqpqgduZMWNGpk6dmhVWWKGuxgX585//nHPOOSdvv/12lltuuay//vp1x9lLcnzs/MM8zj84/wDwVQTjAHxjyy23XN3B4+Jq3bp1WrZsmRtuuGGB76+xxhp1z6z69NNPs/baa9e9N2nSpLz33nsNpqtK5l1lf/7556e6ujovv/xy7r777tx0001ZffXVc9hhh9Vru/zyyyeZ97yy+dOgJcmcOXMyZcqUBR5YL6r52/7Xv/5Vr/b5+/vyVeff1NNPP53q6ur07t17ibfRunXrbL755jnkkEMavNekycL/t2G55ZZLWVlZDj744AUe9P33Ad3S9sMf/jDLLbdcHnzwwbRs2TKrrrpqvekI52vTpk2SeVPf/fcJpyR1J1+WxKKMo/knRv77pMz8A+Yv22WXXbLLLrvks88+y+OPP56rr746xx57bHr27JlOnTotcZ0AAMCy5Zh54eY/77xz58657777svbaa6e8vDyPPfZYHnroocXaVtu2bfPuu+82WL6g461Fseeee+ahhx7KzTffnO233z59+vRJMq9PNtlkkxx33HELXK9Zs2Zp3bp13UUNX76T+G9/+1tqa2vTunXrDBo0KNttt12uvPLKrLbaaikrK8vYsWPrnn+9uJx/mMf5B+cfAL6KqdQBKKhNNtkk06dPT21tbbp161b35+9//3suv/zyzJ07N927d0/Tpk3rphSbb8yYMTnmmGPqnic134MPPphNN900kydPTkVFRXr06JFhw4alTZs2mTRp0gJrSOZNf/Vl9913X6qrqxd4EmFRbbjhhmnWrFnuvffeesufffbZTJo0aaFX/S+uuXPnZtSoUenQocMiPz9tQTbZZJO8+eab6dKlS11fbLDBBrnuuuvyyCOPLHS9Vq1apWvXrnn77bfr9eN6662Xyy67rN7Ueoviq66qX5BmzZplu+22y0MPPZQHHnhgoVdkz5+KbMqUKfXq/PTTT3PJJZfUHSAu7v6TRRtHzZs3T48ePfLwww/Xu0J9/Pjx9dY5+uijM2jQoCTzThbstNNOGThwYObOnZuPP/54sWsDAACKU6kfM7/99tuZOnVqDjzwwKy77rp1x2J/+tOfknz1neb/bdNNN80HH3yQCRMm1C379NNP8+KLLy5xfaecckqaN2+es846K3PmzEky7/t45513stZaa9Xrk7vvvju33357Kioq0qtXr8yZM6fucyTz7gIfOnRorrzyyrzyyiuZNWtWBgwYkNVXX70uPJ8fii/JHc3OPzj/4PwDwNdzxzgABbXVVluld+/eGThwYAYOHJh11lknL7/8ci699NJsueWWddNnHXjggbnuuuvSrFmzbLLJJnnppZdy00035bjjjmtwELHxxhunpqYmgwYNyoABA7LccsvlgQceyGeffdbgmWtJsu6662aPPfbIpZdemhkzZqR379557bXXMnLkyPzgBz/IlltuucSfb4UVVsiAAQNy+eWXp2nTptlmm23ywQcf5JJLLqnb7+J677336g7s58yZkw8++CA333xzXn311Vx++eXf6OrogQMHZt99983hhx+e/fbbL82bN88tt9ySRx99NJdeeulXrnvMMcdkwIAB+eUvf5nddtst1dXVGTNmTF566aUMHDhwsepo06ZNnn/++TzzzDPp1avXIj23rH///jn88MNTXl6ek08+eYFtOnfunN122y2nnHJKPvzww2ywwQZ55513cvHFF2fVVVfNmmuuWbf/f/3rX3nsscfqPcvsqyzqODrmmGNy0EEHZfDgwdlnn33yzjvvZPTo0fW2temmm+a0007L8OHD88Mf/jD//ve/M3LkyKy55ppZf/31F6keAACg+JX6MfNaa62VVq1aZfTo0WnSpEmaNGmShx56KLfffnuSr342+X/70Y9+lBtuuCGDBw/OL37xi7Rq1SpXXHHFYoXr/23VVVfN//7v/2bUqFG5/vrrc9hhh+Xggw/O3XffnYMPPjiHHnpo2rZtm/vvvz+33nprhg4dmiTZeuut06NHj5xwwgk5+uijs9pqq+Xuu+/OW2+9lTPPPDPLL798mjRpkvPPPz+HHnpoZs+enXHjxuWPf/xjkizRM7Cdf3D+wfkHgK8nGAegoMrLy3PVVVflkksuyZVXXplPPvkknTp1yiGHHFJ3xWqSHHvssWnfvn1uvvnmXHPNNVl11VVzyimnZN99922wzRVXXDHXXHNNLrnkkpx00kmZMWNG3ZXDm2666QLrOPvss7PGGmvkjjvuyNVXX50VV1wxBx54YAYOHLhEV+9+2ZAhQ9KhQ4f85je/yS233JIVVlghO+64Y44++uh6z8laVFdccUWuuOKKJPOmF2vXrl169eqVU089daHPOFtU66+/fsaOHZuLL744xx13XGpra/P9738/l19+ebbddtuvXLdPnz659tprM3LkyBx11FFp2rRpqqqq8utf/zobbbTRYtVxxBFHZNSoUfnZz36W+++/f5GmGNt8883Tpk2brLTSSllnnXUW2u7cc8/NlVdemZtvvjkfffRR2rdvn/79++foo4+uu5Nizz33zGOPPZZBgwblqKOOWuTnai3KOOrVq1euvvrqXHTRRRk8eHBWXXXVnHPOOTniiCPqtrPvvvtmzpw5ufnmm/Pb3/42LVq0yGabbZZjjz02TZs2XaRaAACA4lfqx8ytW7fOqFGjMmLEiPz85z/Pcsstly5duuQ3v/lNfvazn+XZZ59N3759F2lbzZo1y/XXX59zzjknZ599dsrKyvLjH/84q622Wj755JMlrnHAgAG56667MmrUqOy2227p1KlTbr755lx44YUZNmxYZs2alTXXXDNnn3129tprryRJRUVFrr766lxwwQW55JJLMmPGjHTu3DljxoxJ9+7dkyQXXnhhRo4cmSOPPDLLL798Ntpoo9x444054IAD8uyzz6Zz586LXavzD84/OP8A8NXKapdkXhYAAAAAAAAAKBKeMQ4AAAAAAABASROMAwAAAAAAAFDSBOMAAAAAAAAAlDTBOAAAAAAAAAAlTTAOAAAAAAAAQEkTjAMAAAAAAABQ0poUuoBlpaamJnPnzk15eXnKysoKXQ4AAAAFVltbm5qamjRp0iTl5a4bb+wc1wMAAPDfFufY/jsTjM+dOzcTJkwodBkAAAA0Mt26dUuzZs0KXQZfw3E9AAAAC7Mox/bfmWB8/hUC3bp1S0VFRYGrYVFVV1dnwoQJ+o1Gz1ilmBivFBPjlWJhrBan+f3mbvHiML+funbt6kKGRs7vxOKgn4qDfioO+qk46Kfioa+Kg34qDt+FflqcY/vvTDA+f5q1ioqKku34UqbfKBbGKsXEeKWYGK8UC2O1OJmWuzg4ri8++qo46KfioJ+Kg34qDvqpeOir4qCfisN3oZ8W5djeZfEAAAAAAAAAlDTBOAAAAAAAAAAlTTAOAAAAAAAAQEn7zjxjHAAAAAAAAOC/VVdXZ86cOYUuY6mrrq5OksycObNonzHetGnTpVa7YBwAAAAAAAD4zqmtrc1HH32UqVOnFrqUb0VtbW2aNGmSd999N2VlZYUuZ4mtsMIK+d73vveNP4NgHAAAAAAAAPjOmR+Kr7jiimnZsmVRh8cLUltbmxkzZqSysrIoP1ttbW2mT5+ejz/+OEmy0korfaPtCcYBAAAAAACA75Tq6uq6ULx9+/aFLudbUVtbm5qamrRo0aIog/EkqaysTJJ8/PHHWXHFFb/RtOrlS6soAAAAAAAAgGIw/5niLVu2LHAlfJ35ffRNnwMvGAcAAAAAAAC+k4r1TurvkqXVR4JxAAAAAAAAAEqaYBwAAAAAAACgCPTt2zedO3eu+1NVVZUdd9wx11133VLf12WXXZYDDjhgqbUrtCaFLgAAAAAAAACgUaitTZbl9OpLsL8TTzwx/fv3T5LMnTs3Tz31VE466aSssMIK2X333ZdaaYceeugiBd6L2q7QBOMAAAAAAAAAybyQ+rW3k+kzv/19tWyRdFl7sVdr3bp1OnbsWPd6jz32yL333puHH354qQbjyy233FJtV2imUgcAAAAAAACYb/rM5PPp3/6fpRi+N2nSJE2bNs0BBxyQM888M9tuu2222WabfPHFF/nHP/6RI444IhtuuGH69u2bkSNHprq6um7dP/3pT9ljjz2y4YYbZrfddsuTTz6ZpP4U6XPmzMnJJ5+cH/zgB+nRo0eOOOKI/POf/2zQLkleeOGF7Lffftloo43St2/f3HTTTXXvnXDCCTn33HNz9NFHZ8MNN8xWW22Vu+66a6l9D19FMA4AAAAAAABQhObMmZOHH344TzzxRLbddtskybhx43L++efnsssuS8uWLTNkyJC0b98+d955Z84999z87ne/y+jRo5Mkb7zxRo488shsv/32ufvuu7PLLrtk4MCBmTx5cr39jB07Ns8880zGjBmT22+/PV988UXOOeecBvW89dZbOeigg9K7d++MGzcuQ4YMyfDhw/PII4/U21ZVVVXuvffe9OvXL6eddlo+++yzb/FbmsdU6gAAAAAAAABF4rTTTsuZZ56ZJJk5c2ZatGiRgw46KLvttltuu+22bL311tl4441TW1ubP/zhD5k0aVJuu+22lJeXZ+21187xxx+foUOHZtCgQbn99tuz8cYbZ+DAgUmSAQMGZPr06fn3v/9db58ffPBBmjdvnlVWWSUrrLBCzjvvvEydOrVBbbfeemu6du2aY445Jkmy9tpr56233so111yT7bffPknSuXPn/OxnP0uS/PznP88NN9yQN954IxtvvPG39ZUlEYwDAAAAAAAAFI2jjjoq/fr1S5I0b948HTt2TEVFRd37q6yySt3P77zzTqZOnZqePXvWLaupqcnMmTMzZcqUvPPOO6mqqqq3/aOPPrrBPvfZZ5/cd9996dOnTzbZZJNst9122XPPPRu0e+utt9K9e/d6y3r06JGbb7657vWaa65Z93OrVq2SJHPnzl2ET/7NCMYBAAAAAAAAikT79u2zxhprLPT95s2b1/1cXV2dtddeO6NGjWrQrnXr1mnSZNHi4vXWWy/jx4/PH//4x/zxj3/MRRddlHvvvTdjx45d6L7nq6mpqfdM86ZNmzZoU1tbu0h1fBOCcQAAAAAAAIAStOaaa2bSpElp165dWrdunSR54oknMm7cuIwYMSJrrLFGXnvttXrr7LvvvjnggAPqLbvrrrvSrFmz9O/fPzvttFNefPHF7LPPPvnkk0/qtVtrrbXyzDPP1Fv2wgsvZK211voWPt3iKS90AQAAAAAAAAAsfZtuumlWWWWVHHvssZk4cWKeffbZnHLKKamsrExFRUX222+/PPvss/n1r3+dd999N1deeWXeeOON9OrVq952Pvvss5x99tl58skn8/777+d3v/tdvve976Vt27b12v3kJz/Ja6+9losuuijvvPNO7rzzzvz2t7/N/vvvvyw/9gK5YxwAAAAAAABgvpYtSmY/FRUVGTVqVM4666z8+Mc/TsuWLbPjjjvm+OOPT5Ksvvrqueyyy3LhhRfmoosuynrrrZfRo0enU6dO9baz//7756OPPsqxxx6badOmZYMNNsgVV1xR79nmSbLyyivnyiuvzIgRIzJmzJisvPLKOeGEE/I///M/3/pn/TqCcQAAAAAAAIAkqa1Nuqy9bPdXVrbIzcePH/+V7994440Nlq222mq56qqrFrrONttsk2222abB8iFDhtT9XF5enmOPPTbHHnvsV7ZLks022yx33nnnAvd13nnnNVg2ceLEhda2NJlKHQAAAAAAACBZrJC6KPf3HSYYBwAAAAAAAKCkCcYBAAAAAAAAKGmCcQAAAAAAAABKmmAcAAAAAAAAgJImGAcAAABgqaqsrCx0CSwC/VQc9BMAwNLRpNAFAAAAlAInrWHZqKioKHQJfI2Kiop07dq10GXwNfRTcdBPDdVU16S8wv1eAMDiE4wDAABLpLamJmXlTkomTlovjDHCt+Gew+7Jxy99XOgyACiAjl06Zs+xexa6DACgSAnGAQCAJVJWXp7q8eNTO3VKoUuhESpboW0q+vYtdBmUoH9N/Fc+euGjQpcBAABAkRGMAwAAS6x26pTkk08KXQaNUG2hCwAAAIAS1Ldv33z44Yd1r8vKytKmTZv07Nkzp556alZaaaVvbd8nnHBCkuS8887LZZddlqeffjo33njjt7a/pc2cdgAAAAAAAABJaqprGv3+TjzxxDz++ON5/PHH89hjj+Xiiy/OG2+8keOPP/5bqLB0uGMcAAAAAAAAIEl5RXnG7T8uk1+b/K3vq2OXjtlz7J6LvV7r1q3TsWPHutedOnXKUUcdlWOPPTafffZZWrduvTTLLBkFDcZnzZqV008/PQ8//HBatGiRQw89NIceeugC2z7yyCO56KKL8tFHH2X99dfPySefnKqqqmVcMQAAAAAAAFDKJr82OR+98FGhy1gszZo1S5KUl5fn3//+d84888z8/ve/T2VlZXbYYYccd9xxadGiRZLk5Zdfzrnnnpu//e1v+d73vpejjjoqO++8c5Lktttuy7XXXpsPPvggyy23XPr375+TTz45FRUVBftsS0tBp1IfMWJEXnnllVx//fU57bTTMnLkyDz44IMN2r3xxhv55S9/mcMPPzx33313unTpksMPPzwzZswoQNUAAAAAAAAAjcN7772Xq666KltuuWWWW265nHTSSfnss8/y29/+NhdddFFeeeWVnHHGGUmSTz75JIceemi6dOmSO++8M4cffniOP/74vP7663n66adz1lln5ZhjjsmDDz6Y008/Pbfffnt+//vfF/gTLh0Fu2N8+vTpue2223L11VenqqoqVVVVeeONNzJ27NjsuOOO9do+8cQTWXfddbP77rsnSY455piMHTs2b775Zrp161aA6gEAAAAAAACWvdNOOy1nnnlmkmTu3Llp2rRptt1225x44ol577338uijj+bpp59Oq1atMn369JxxxhnZY489MnTo0Nx3331Zfvnlc/LJJ6e8vDxrr712pk2blpkzZ6Zly5Y5++yz069fvyTJqquuml//+td544036pYVs4IF46+//nrmzp2bHj161C3r2bNnRo8enZqampSX/+dm9hVWWCFvvvlmnnvuufTo0SPjxo1Lq1atsvrqqxeidAAAAAAAAICCOOqoo9KvX7988cUXueyyy/Lhhx/ml7/8Zdq2bZsXX3wxNTU1+eEPf5gkqa2tTVlZWWpqavLuu+/mnXfeSdeuXetlsYccckjdzy1atMill16aN998MxMnTsy7776bPn36LPPP+G0oWDA+efLktG3btm6++yTp0KFDZs2alalTp6Zdu3Z1y/v375/x48fnJz/5SSoqKlJeXp4rr7wyyy+//GLvt7q6eqnUz7Ixv7/0G42dsUoxMV4pJsZr41YKz5bi29eY//425toAAABgYdq3b5811lgjSXLJJZdkr732ysCBA3PLLbekuro6rVu3zh133JHa2trMmDEjlZWVKSsrS6dOndKkycLj4T//+c8ZNGhQdt9992y55ZYZNGhQTj/99GX1sb51BQvGZ8yYUS8UT/7zUPjZs2fXWz5lypRMnjw5p556ajbccMPcdNNNGTp0aO688860b99+sfY7YcKEb1Y4BaHfKBbGKsXEeKWYGK+NT2VlZbp27VroMigCEydOzIwZMwpdBgAAAJSkZs2a5ayzzso+++yT6667Ln379s1nn32WsrKyrL766pk+fXref//9XHbZZTn33HOz5ppr5rHHHqu7kzxJjj766GywwQZ5+eWX8z//8z857bTTksybpv29997LpptuWsiPuNQULBhv3rx5gwB8/usWLVrUW37BBRfk+9//fvbff/8kyZlnnpmddtopd9xxRwYMGLBY++3WrZs7W4pIdXV1JkyYoN9o9IxVionxSjExXqH4de7cudAlLNT83zEAAABQzLp375699toro0aNym677ZYtt9wyv/rVr3LyySdn9uzZOfvss7PCCiukTZs22XXXXXPJJZdkxIgR2WefffL888/n97//fQ4//PC89957eeGFFzJx4sS6GbwnT57cINMtVgULxjt16pQpU6Zk7ty5dbfsT548OS1atEibNm3qtX311VdzwAEH1L0uLy/P+uuvn0mTJi32fisqKpxULUL6jWJhrFJMjFeKifEKxcvfXQAAAIpNxy4di24/v/jFL/LQQw/l/PPPz4gRI3LWWWfl4IMPTkVFRbbccsuccsopSZI2bdrkyiuvzDnnnJMbb7wxq622Wi688MJ06dIlgwcPztChQ7PPPvukVatW2WqrrbLffvvltddeW2p1FlLBgvEuXbqkSZMmefHFF9OrV68kyXPPPZdu3brVe9h7kqy44op566236i1755130q1bt2VWLwAAAAAAAFDaaqprsufYPZfp/soryr++4f9v/PjxC1zerl27PP3003WvL7rootTW1mb69Olp2bJl3bTpSdKjR4/cdtttDbax4oor5tprr13ovs8777y6n4cMGbLINTcWi/4tL2WVlZXZfffdM2zYsLz88st59NFHM2bMmBx44IFJ5t09PnPmzCTJj3/849x6662566678u677+aCCy7IpEmTssceexSqfAAAAAAAAKDELE5IXYz7+y4r2B3jSTJ06NAMGzYsBx10UFq1apUhQ4akX79+SZI+ffrk3HPPzZ577pn+/fvniy++yJVXXpmPPvooXbp0yfXXX5/27dsXsnwAAAAAAAAAikBBg/HKysoMHz48w4cPb/DexIkT673ee++9s/feey+r0gAAAAAAAAAoEe7NBwAAAAAAAKCkCcYBAAAAAAAAKGmCcQAAAAAAAOA7qaamptAl8DWWVh8V9BnjAAAAAAAAAMtas2bNUl5enkmTJqVjx45p1qxZysrKCl3WUlVbW5tZs2alvLy8KD9bbW1tZs+encmTJ6e8vDzNmjX7RtsTjAMAAAAAAADfKeXl5VlrrbXyj3/8I5MmTSp0Od+K2trazJkzJ02bNi3KYHy+li1bZvXVV095+TebDF0wDgAAAAAAAHznNGvWLKuvvnrmzp2b6urqQpez1FVXV+f111/Puuuum4qKikKXs0QqKirSpEmTpRLsC8YBAAAAAACA76SysrI0bdo0TZs2LXQpS938sL9FixZFG4wvTd/sfnMAAAAAAAAAaOQE4wAAAAAAAACUNME4AAAAAAAAACVNMA4AAAAAAABASROMAwAAAAAAAFDSBOMAAAAAAAAAlDTBOAAAAAAAAAAlTTAOAAAAAAAAQEkTjAMAAAAAAABQ0gTjAAAAAAAAAJQ0wTgAAAAAAAAAJU0wDgAAAAAAAEBJE4wDAAAAAAAAUNIE4wAAAAAAAACUNME4AAAAAAAAACVNMA4AAAAAAABASROMAwAAAAAAAFDSBOMAAAAAAAAAlDTBOAAAAAAAAAAlTTAOAAAAAAAAQEkTjAMAAAAAAABQ0gTjAAAAAAAAAJQ0wTgAAAAAAAAAJU0wDgAAAAAAAEBJE4wDAAAAAAAAUNIE4wAAAAAAAACUNME4AAAAAAAAACVNMA4AAAAAAABASROMAwAAAAAAAFDSBOMAAAAAAAAAlDTBOAAAAAAAAAAlTTAOAAAAAAAAQEkTjAMAAAAAAABQ0gTjAAAAAAAAAJQ0wTgAAAAAAAAAJU0wDgAAAAAAAEBJE4wDAAAAAAAAUNIE4wAAAAAAAACUNME4AAAAAAAAACVNMA4AAAAAAABASROMAwAAAAAAAFDSBOMAAAAAAAAAlDTBOAAAAAAAAAAlTTAOAAAAAAAAQEkTjAMAAAAAAABQ0gTjAAAAAAAAAJQ0wTgAAAAAAAAAJU0wDgAAAAAAAEBJE4wDAAAAAAAAUNIE4wAAAAAAAACUNME4AAAAAAAAACVNMA4AAAAAAABASROMAwAAAAAAAFDSBOMAAAAAAAAAlDTBOAAAAAAAAAAlTTAOAAAAjVDfvn0zbty4BsvHjRuXvn375phjjslWW22VGTNmNGhzyCGHZN99901tbW2SpKamJtdff3122223bLjhhtlmm21y1llnZerUqfXWq62tzdixY7PrrrumW7du6dOnT0444YS8//77DWrr3LlznnnmmQb7/tOf/pTOnTvnhBNOqKu3c+fOdX+qqqqy44475q677lrCbwYAAAAWn2AcAAAAitDxxx+fzz77LKNHj663/OGHH84zzzyTM844I2VlZUmSn//857n++utzxBFH5N577815552X559/PocddlhmzZpVt+6JJ56Yyy+/PAcffHAeeOCBjBw5Mp9//nn23nvvTJw4sd5+mjZtmvHjxzeo69FHH63b73zf+9738vjjj+fxxx/PQw89lAEDBuTkk0/OCy+8sLS+DgAAAPhKgnEAAAAoQp06dcqQIUPy61//uu6O7pkzZ+a8887LIYccku9///tJknvuuSd/+MMfct1116V///5ZbbXV8oMf/CBXXXVV3nzzzdx9991J5gXa9957b66//vr8z//8T1ZdddVstNFGueyyy7LxxhvnxBNPrLf/Xr16NQjGa2trM378+Gy00Ub1lldUVKRjx47p2LFjVl111ey55575wQ9+kAcffPBb+nYAAACgPsE4AAAAFKkDDjgga6yxRs4///wkyTXXXJPy8vIMGjSors2dd96Z7bffPquvvnq9dTt06JDrr78+/fr1S5Lceuut6du3b9Zbb7167crKyjJw4MC88soree211+qWb7311vnggw/y1ltv1S178cUXs/zyy2fNNdf82tpbtmy52J8XAAAAllRBg/FZs2blxBNPTK9evdKnT5+MGTNmge0OOOCAes8jm/9n6NChy7hiAAAAaDyaNGmSU089NQ8//HAeffTRXHvttTnttNPSokWLujavv/56unXrtsD1N9xww6ywwgpJkldeeWWh7aqqqlJZWZmXX365blmbNm3Ss2fPeneNP/LII9luu+2+tu7nnnsuf/nLX7LzzjsvyscEAACAb6xJIXc+YsSIvPLKK7n++uszadKkHH/88Vl55ZWz44471mt32WWXZc6cOXWvX3rppRx99NH5yU9+sqxLBgAAgGXmtNNOy5lnnllv2dy5c9OxY8e61717986uu+6an//859lhhx2y5ZZb1mv/2Wef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mztbikh1dXUmTJig32j0jFWKifFKMTFeofh17ty50CUs1PzfMQAAAEDpK1gw3qVLlzRp0iQvvvhievXqlSR57rnn0q1bt5SX13/0+QknnJCysrKce+65dctef/31fP/731/s/VZUVDipWoT0G8XCWKWYGK8UE+MVipe/uwAAAEBjUP71Tb4dlZWV2X333TNs2LC8/PLLefTRRzNmzJgceOCBSebdPT5z5swkSd++ffO73/0ud911V959992MHDkyzz33XH76058WqnwAAAAAAAAAikTBgvEkGTp0aKqqqnLQQQfl9NNPz5AhQ9KvX78kSZ8+fXL//fcnSfr165fTTjstV1xxRXbZZZeMHz8+11xzTVZdddVClg8AAAAAAABAESjYVOrJvLvGhw8fnuHDhzd4b+LEifVe77333tl7772XVWkAAAAAAAAAlIiC3jEOAAAAAAAAAN82wTgAAAAAAAAAJU0wDgAAAAAAAEBJE4wDAAAAAAAAUNIE4wAAAAAAAACUNME4AAAAAAAAACVNMA4AAAAAAABASROMAwAAAAAAAFDSBOMAAAAAAAAAlDTBOAAAAAAAAAAlTTAOAAAAAAAAQEkTjAMAAAAAAABQ0gTjAAAAAAAAAJQ0wTgAAAAAAAAAJU0wDgAAAAAAAEBJE4wDAAAAAAAAUNIE4wAAAAAAAACUNME4AAAAAAAAACVNMA4AAAAAAABASROMAwAAAAAAAFDSBOMAAAAAAAAAlDTBOAAAAAAAAAAlTTAOAAAAAAAAQEkTjAMAAAAAAABQ0gTjAAAAAAAAAJQ0wTgAAAAAAAAAJa3Jkq74+eef580338zcuXNTW1tb773evXt/48IAAAAAAAAAYGlYomD87rvvzrBhwzJjxowG75WVleW11177xoUBAAAAAAAAwNKwRMH4xRdfnL333jtHHXVUWrVqtbRrAgAAAAAAAIClZomeMT516tQceOCBQnEAAAAAAAAAGr0lumN8m222ycMPP5xDDz10adcDAAAAReGuu+5a5La77777t1YHAAAA8PWWKBjv1KlTLr744jzwwANZY4010rRp03rvn3vuuUulOAAAAGisLr300nqv//GPf6RZs2ZZbbXV0rRp07z77ruZNWtW1l9/fcE4AAAAFNgSBePTpk3LLrvssrRrAQAAgKIxfvz4up+vuOKKTJgwIeecc05WWGGFJMnnn3+eU089NR06dChQhQAAAMB8SxSMuyMcAAAA/uPaa6/NLbfcUheKJ0mrVq0yePDg7LXXXjnxxBMLVxwAAACQ8iVd8dFHH82+++6bTTbZJD179sxee+21WM9XAwAAgFLRunXr/O1vf2uw/Lnnnku7du0KUBEAAADwZUt0x/jNN9+c4cOH56c//WkGDBiQmpqaPP/88zn99NMzZ86c7L333ku7TgAAAGi0Dj/88Jx00kn561//mi5duqS2tjYTJkzIAw88YNY1AAAAaASWKBi/5pprctppp2X33XevW7bddttlvfXWy+jRowXjAAAAfKfsu+++WWWVVXL77bfnpptuSpKst956GTNmTHr16lXg6gAAAIAlCsY/+eSTbLTRRg2W9+jRI//4xz++aU0AAABQdLbccstsueWW+fzzz1NdXZ3ll1++0CUBAAAA/78lesZ4ly5dFvg88TvvvDPrrrvuN60JAAAAis7111+fLbfcMr17986mm26aLbbYIiNHjix0WQAAAECW8I7xY489NgcffHD++te/ZsMNN0ySvPjii3n99dczevTopVogAAAANHaXX355fvOb3+TnP/95evTokZqamjz//PMZOXJkmjVrlgEDBhS6RAAAAPhOW6JgvEePHhk3blxuvfXWvPXWW2nevHl69+6diy++OCuttNLSrhEAAAAatVtvvTVnn312+vbtW7esS5cu6dSpU84++2zBOAAAABTYEgXjSbLOOutk6NChS7MWAAAAKEqff/551lxzzQbL11prrXz66afLviAAAACgnkUOxg888MCMHDkybdq0yQEHHJCysrKFtr3hhhuWSnEAAABQDHr06JExY8bkjDPOSHl5eZKkuro6Y8aMSffu3QtcHQAAALDIwfgmm2ySpk2bJkl+8IMffGsFAQAAQLEZOnRo9t9///zlL39JVVVVkuTVV1/N7Nmzc8011xS4OgAAAGCRg/HBgwcv8Of5Pv3007Rt2/Yr7yQHAACAUrTOOuvkgQceyO9+97u8/fbbad68ebbYYovsuuuuWW655QpdHgAAAHznLdEzxv/5z3/mvPPOy4ABA7L22mvnf//3f/Pcc89lpZVWyqhRo7L++usv7ToBAACgUWvbtm0OPPDAQpeRJDnhhBNy5513LvT9TTbZJEly44031i2bPXt2BgwYkP/7v//Lb3/726y88spJktdeey1XXnllnn322UydOjUrr7xy+vfvnyOOOCItWrSot93p06dn8803T9euXfPb3/62bnnnzp0XWssqq6yS8ePHL9HnBAAAgEW1RMH4sGHDMn369KywwgoZN25c/v73v+fmm2/OPffckzPPPDNjx45d2nUCAABAo/WPf/wjF1xwQV5//fXMmjUrtbW19d7//e9/v0zrOemkk/LLX/4ySXL//fdnzJgxuf322+vev/HGG/PCCy/Uva6pqclxxx2Xv//97xk7dmxdKP7EE0/kyCOPTL9+/XLFFVekffv2ee2113LhhRfm9ddfz+jRo+vtd/z48enYsWOef/75vP/++1lttdWSJI8//nhdmyFDhqRHjx459NBDkyQVFRXfzpcAAAAAX7JEwfhTTz2VcePGZaWVVsqjjz6abbfdNhtuuGHatWuXXXbZZWnXCAAAAI3acccdl2nTpmWfffZJ69atC11OWrduXVdH69atU1FRkY4dO9a937Rp03rtzz777Dz++OO58cYbs9ZaayWZdwf5SSedlD322COnn356XduVV145nTt3zg477JBXXnklG2ywQd179957b7bbbrs88cQTueuuuzJkyJAkabDvli1b1lsGAAAA37YlCsabN2+eWbNmZdq0afnrX/+aCy+8MEnywQcfZPnll1+qBQIAAEBj99JLL+WOO+7IeuutV+hSFtuoUaNyxx135Nprr02XLl3qlj/++OP55z//maOOOqrBOquuumoefPDBujvCk2TatGl5/PHHs/fee6dZs2a56667Mnjw4JSVlS2TzwEAAABfpXxJVtpuu+1y9NFH56CDDsryyy+frbfeOvfff3+OPfbY/OhHP1raNQIAAECjtsYaa2TatGmFLmOx3Xbbbbnkkkvys5/9LD179qz33ksvvZQ111wz7du3X+C6Xw7Fk+Thhx9ORUVFNt9882y77bb54IMP8uyzz35rtQMAAMDiWOJnjP/mN7/Jhx9+mH322SfNmzfP7Nmzc8QRR2T//fdf2jUCAABAo/PMM8/U/bzTTjvluOOOy5FHHpnVVlutwXOze/fuvazL+1pvvPFGTj/99PTs2TNjx47Nfvvtl3bt2tW9P2XKlAazwp1wwgl56KGH6l4ffvjhOeKII5Ik9913XzbffPNUVlamW7du+d73vpc777xzqX/2Dp07pGZmzVLdZqnq2GXedPXV1dXLdL81NTWprKxMTY1+asz0U3HQT8Vh/u/ZZf37lsWjn4qHvioO+qk4fBf6aXE+2xIF402aNMnBBx9cb9nuu+++JJsCAACAonTAAQc0WHbKKac0WFZWVpbXXnttWZS0WKZMmZKzzjorO+20U/r3758zzzwzF198cd37bdq0yWeffVZvnV/96lc58sgj636eM2dOkmTy5Ml5+umnc+aZZyaZ95m33377jBs3LqecckoqKyuXWt27XbNbgwsPWLia6ppl/n1VVFSka9euy3SfLD79VBz004LVVFfnlVdfrft3qLGYMGFCoUtgEein4qGvioN+Kg76aZ5FDsYPPPDAjBw5Mm3atMkBBxzwlc8Iu+GGG5ZKcQAAANBYvf7664Uu4RvZeOONs/feeydJTj311AwaNCg77bRT+vXrlyTZcMMNM2bMmEydOjUrrLBCkqRDhw7p0KFDkqRFixZ123rggQdSXV2dU045pe7igNra2tTU1OSRRx7JbrvttvQKf/fDZObcpbe9Elc+d24yx/cFlJCWLVLeZe1UVVUVupI61dXVmTBhQrp16+birUZMPxUPfVUc9FNx+C700/zPuCgWORjfZJNN0rRp0yTJD37wgyWrDAAAAErQtttumzvuuKMuQJ7vn//8Z3bfffc8+eSThSnsKzRp8p9TAtttt1122GGHDBs2LL169Uq7du3ywx/+MCuuuGJGjx6dE044od66s2fPzpQpU+pe33///dlss81y4okn1ms3aNCg3HXXXUs3GJ/yWfL5jKW3PQCKUmM8uV9RUdEo66I+/VQ89FVx0E/FQT/Ns8jB+ODBg+v9/Mknn+Tf//531lprrSTzDoJ79+6djh07Lv0qAQAAoJF58MEH89hjjyVJPvzww5xxxhlp3rx5vTYffvhh0Zx8OPnkk+tNqd68efOMGDEiRxxxRKZNm5Yf//jH6dixY1577bWMGjUq7733XqqqqvLBBx/khRdeyCWXXJLvf//79ba5zz775MILL8w///nPdOrUqUCfDAAAAJLyJVnpySefzPbbb5/f/e53dctuuOGG9O/fP88999xSKw4AAAAaq0022aTe69ra2gZt1ltvvYwaNWpZlfSNrLjiijn22GNz//335+GHH04y7zPecccdSZKjjz46O+64Y84999x079499957b/r27Zv7778/bdu2Td++fRtsc88990yTJk1y9913L9PPAgAAAP9tke8Y/7Lhw4fniCOOyIABA+qW3Xzzzbnyyitzzjnn1B00AwAAQKlq165dzj333CTJKquskkMPPTQtW7YscFUN7bnnntlzzz3rLRsyZMgC2+6zzz7ZZ5996i1ba6216j7nggwYMKDe+YEva9euXYNnvd14442LUjYAAAAsVUt0x/j//d//Zccdd2ywfKeddsqbb775jYsCAACAYjJ48OBUV1dn7NixOfvss/Ppp5/mD3/4Q957771ClwYAAABkCYPxtddeOw888ECD5ePHj8/qq6/+jYsCAACAYvL3v/89/fr1yx133JGbbropX3zxRR5++OH86Ec/ytNPP13o8gAAAOA7b4mmUj/66KMzcODAPPHEE6mqqkqSTJw4Mc8++2wuu+yypVogAAAANHZnnXVW9ttvvxx11FHp0aNHkuTcc89Nu3btMmLEiNx+++0FrhAAAAC+25bojvEf/vCHufPOO9OlS5e8/fbbee+997L++uvnvvvuy1ZbbbW0awQAAIBGbcKECdl9990bLN933309cgwAAAAagSW6YzxJ1ltvvQwdOjTTpk1Lq1atUl5enrKysqVZGwAAABSFdu3a5Z133mnweLHnn38+7du3L1BVAAAAwHxLdMd4bW1trrjiivzgBz/IZpttlkmTJuXYY4/NqaeemtmzZy/tGgEAAKBR+9nPfpaTTz45Y8eOTW1tbZ566qlceumlOeOMM3LIIYcUujwAAAD4zluiYPzyyy/PPffck/POOy/NmjVLkuyxxx554oknMmLEiKVaIAAAADR2++67b04//fTcf//9adGiRUaMGJHHH388Z555Zn76058WujwAAAD4zluiqdTvvPPOnHfeeendu3fd9OlbbLFFhg8fnp///Oc5+eSTl2qRAAAA0Nj17ds3ffv2LXQZAAAAwAIsUTD+ySefZMUVV2ywvE2bNpk+ffo3LgoAAAAau5EjRy5y28GDB3+LlQAAAABfZ4mC8U033TTXXnttzjjjjLpln3/+eS666KL84Ac/WOTtzJo1K6effnoefvjhtGjRIoceemgOPfTQr1zngw8+yK677prRo0cv1r4AAABgaRo5cmTKy8vTpUuXLLfccqmtrV1gu/kzrQEAAACFs0TB+LBhwzJ48OBsscUWmTVrVgYOHJhJkyZl5ZVXzhVXXLHI2xkxYkReeeWVXH/99Zk0aVKOP/74rLzyytlxxx2/ct/uSgcAAKDQTjvttDz66KN58cUX07t372y77bbZdttt065du0KXBgAAAPyXJQrG27Rpk9tvvz1PPvlk3n777cydOzdrrbVW+vTpk/Ly8kXaxvTp03Pbbbfl6quvTlVVVaqqqvLGG29k7NixCw3G77nnnnzxxRdLUjIAAAAsVfvtt1/222+/fP7553nsscfyyCOP5Pzzz8/3v//9bLfddtl+++2zyiqrFLpMAAAAIEsYjO+yyy4ZOXJkNttss2y22WZLtOPXX389c+fOTY8ePeqW9ezZM6NHj05NTU2DgH3KlCk5//zzM2bMmOyyyy5LtE8AAABY2lq1apWdd945O++8c2bPnp0nn3wyv//977PvvvumQ4cO2W677TJo0KBClwkAAADfaYt2e/d/r1Renjlz5nyjHU+ePDlt27ZNs2bN6pZ16NAhs2bNytSpUxu0P++887LHHntkvfXW+0b7BQAAgG9Ls2bNsuWWW2bXXXfNzjvvnPfeey9XX311ocsCAACA77wlumN86623ziGHHJJtttkmq6yySr1wO0kGDx78tduYMWNGg/Xmv549e3a95X/5y1/y3HPP5d57712Scuuprq7+xttg2ampqUllZWVqamoKXQp8pfm/W/yOoRgYrxQT47Vxq6ioKHQJFIHG/Pd3adb2xRdf5M9//nPGjx+fP/3pT0nmHTufe+656dOnz1LbDwAAALBkligYnzhxYqqqqvLxxx/n448/rvdeWVnZIm2jefPmDQLw+a9btGhRt2zmzJk59dRTc9ppp9VbvqQmTJjwjbfxbWnatGmqqjZIRcUS3chfkioqKtK1a9dCl9GoVFfX5NVXX/nGszbw7WjMv2PgvxmvFBPjtfGprKz0/2kskokTJ2bGjBmFLuNb8dFHH+X3v/99xo8fn2eeeSadOnVK3759c+mll6Znz54uHgEAAIBGZLGC8bvvvjuPPPJIOnTokG233fYbPeu7U6dOmTJlSubOnZsmTeaVMXny5LRo0SJt2rSpa/fyyy/n/fffz1FHHVVv/Z/97GfZfffdc8YZZyzWfrt169aoT05UVJTnL//3Sf49c26hS6ERatOiSTZfs32qqqoKXQr/pbq6OhMmTGj0v2MgMV4pLsYrFL/OnTsXuoSFmv87Zklts802adKk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"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "# Convert dictionary to DataFrame\n",
    "df_precision = {\n",
    "    \"Method\": [\"NLP_ML\", \"SYN_TAG\", \"TRDM\", \"TKGAT\", \"HYCOMB\"],\n",
    "    \"Precision\": [0.400, 0.314, 0.035, 0.365, 0.813],\n",
    "    \"Recall\": [None, 0.162, None, 0.072, 0.364 ],\n",
    "}\n",
    "\n",
    "df_combined = pd.DataFrame({\n",
    "    \"Method\": [\"SYN_TAG\", \"TKGAT\", \"HYCOMB\"],\n",
    "    \"Precision\": [0.314, 0.365, 0.813],\n",
    "    \"Recall\": [0.162, 0.072, 0.364],\n",
    "})\n",
    "\n",
    "# Create a figure with 1 row and 2 columns of subplots\n",
    "plt.figure(figsize=(20, 8))\n",
    "\n",
    "# Precision Bar Chart\n",
    "plt.subplot(1, 2, 1)  # 1 row, 2 columns, subplot 1\n",
    "plt.bar(df_precision['Method'], df_precision['Precision'], color=plt.cm.Paired(range(5)))\n",
    "plt.xlabel('Method')\n",
    "plt.ylabel('Precision')\n",
    "plt.title('Precision of Different Methods')\n",
    "plt.xticks(rotation=45)  # Rotate method names if necessary\n",
    "\n",
    "# Combined Precision and Recall Horizontal Bar Chart\n",
    "plt.subplot(1, 2, 2)  # 1 row, 2 columns, subplot 2\n",
    "\n",
    "pos = list(range(len(df_combined['Method'])))\n",
    "width = 0.4\n",
    "\n",
    "# Plotting horizontal bars\n",
    "for i, p in enumerate(pos):\n",
    "    plt.barh(p - width / 2, df_combined.iloc[i]['Precision'], height=width, color='pink', label='Precision' if i == 0 else \"\")\n",
    "    plt.barh(p + width / 2, df_combined.iloc[i]['Recall'], height=width, color='purple', label='Recall' if i == 0 else \"\")\n",
    "\n",
    "plt.xlabel('Score')\n",
    "plt.ylabel('Method')\n",
    "plt.title('Precision and Recall of Different Methods')\n",
    "plt.yticks(pos, df_combined['Method'])\n",
    "plt.legend()\n",
    "\n",
    "# Adjust layout to prevent overlap\n",
    "plt.tight_layout()\n",
    "\n",
    "# Show the plots\n",
    "plt.show()"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2024-09-14T05:12:05.189748Z",
     "start_time": "2024-09-14T05:12:04.762955Z"
    }
   },
   "id": "1b7cef027d744c6e",
   "execution_count": 7
  },
  {
   "cell_type": "markdown",
   "source": [],
   "metadata": {
    "collapsed": false
   },
   "id": "e3e166e776371acf"
  }
 ],
 "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
}
