{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/vibhhus/anaconda3/envs/py374/lib/python3.7/site-packages/tqdm/auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "from tqdm import tqdm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "user_vectors = np.load('uservectors.npy')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "item_vectors = np.load('itemvectors.npy')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_items = len(item_vectors)\n",
    "n_users = len(user_vectors)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_recommendations(user_vector, already_rated):\n",
    "    predicted_ratings = {index: None for index in range(0, n_items)}\n",
    "    for item in range(0, n_items):\n",
    "        item_rating = user_vector @ torch.tensor(item_vectors[item])\n",
    "        predicted_ratings[item] = item_rating\n",
    "    sorted_pred_ratings = sorted(predicted_ratings.items(), key=lambda x: x[1], reverse=True)\n",
    "\n",
    "    count_var = 0\n",
    "    top_10_item_names=[]\n",
    "    for i in sorted_pred_ratings:\n",
    "        item_id = i[0]\n",
    "        if item_id not in already_rated:\n",
    "            count_var+=1\n",
    "            top_10_item_names.append(item_id)\n",
    "        if count_var==10:\n",
    "            break\n",
    "    return top_10_item_names"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_recommendation_scores(user_vector, already_rated):\n",
    "    predicted_ratings = {index: None for index in range(0, n_items)}\n",
    "    for item in range(0, n_items):\n",
    "        item_rating = user_vector @ torch.tensor(item_vectors[item])\n",
    "        predicted_ratings[item] = item_rating\n",
    "    sorted_pred_ratings = sorted(predicted_ratings.items(), key=lambda x: x[1], reverse=True)\n",
    "    count_var = 0\n",
    "    top_10_item_names=[]\n",
    "    top_10_item_scores=torch.tensor([1, 2, 3, 4, 5, 6, 7, 8, 9, 10], dtype=torch.float64)\n",
    "    for i in sorted_pred_ratings:\n",
    "        item_id = i[0]\n",
    "        item_score = i[1]\n",
    "        if item_id not in already_rated:\n",
    "            top_10_item_scores[count_var]=item_score\n",
    "            count_var+=1\n",
    "            top_10_item_names.append(item_id)\n",
    "            # top_10_item_scores.append(item_score)\n",
    "        if count_var==10:\n",
    "            break\n",
    "    return top_10_item_names, top_10_item_scores"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "#define user_type.choice\n",
    "#define user_type.rating\n",
    "import random\n",
    "class UserType:\n",
    "    def __init__(self, user_type, rec_scores, popularities, beta=0):\n",
    "        self.user_type = user_type\n",
    "        self.rec_scores = rec_scores\n",
    "        self.popularities = popularities\n",
    "        self.beta = beta\n",
    "        self.set_user_type_values()\n",
    "\n",
    "    def set_user_type_values(self):\n",
    "        if self.user_type == \"Enjoyer\":\n",
    "            self.item_choice = 0\n",
    "            self.item_rating = 5\n",
    "        elif self.user_type == \"Hater\":\n",
    "            self.item_choice = 0\n",
    "            self.item_rating = 1\n",
    "        elif self.user_type == \"Random Enjoyer\":\n",
    "            self.item_choice = random.randint(0, len(self.rec_scores) - 1)\n",
    "            self.item_rating = 5\n",
    "        elif self.user_type == \"Random Hater\":\n",
    "            self.item_choice = random.randint(0, len(self.rec_scores) - 1)\n",
    "            self.item_rating = 1\n",
    "        elif self.user_type == \"Choice Enjoyer\":\n",
    "            # probabilities = np.exp(self.beta * np.array(self.rec_scores))\n",
    "            # probabilities /= probabilities.sum() \n",
    "            # self.item_choice = np.random.choice(len(self.rec_scores), p=probabilities)\n",
    "            # self.item_rating = 5\n",
    "            \n",
    "            probabilities = torch.softmax(torch.mul(self.rec_scores, self.beta), dim=0)\n",
    "\n",
    "            # Choose an item based on probabilities\n",
    "            self.item_choice = torch.multinomial(probabilities, 1).item()\n",
    "\n",
    "            # Set item rating to 5\n",
    "            self.item_rating = torch.tensor(5, dtype=torch.float64)\n",
    "        elif self.user_type == \"Choice Hater\":\n",
    "            probabilities = np.exp(self.beta * np.array(self.rec_scores))\n",
    "            probabilities /= probabilities.sum() \n",
    "            self.item_choice = np.random.choice(len(self.rec_scores), p=probabilities)\n",
    "            self.item_rating = 1\n",
    "        elif self.user_type == \"Popular Enjoyer\":\n",
    "            probabilities = np.exp(self.beta * np.array(self.rec_scores))\n",
    "            probabilities /= probabilities.sum() \n",
    "            self.item_choice = np.random.choice(len(self.values), p=probabilities)\n",
    "            if self.popularities[self.item_choice]<500:\n",
    "                self.item_rating = 5\n",
    "            else:\n",
    "                self.item_rating = 1\n",
    "        elif self.user_type == \"Niche Enjoyer\":\n",
    "            probabilities = np.exp(self.beta * np.array(self.rec_scores))\n",
    "            probabilities /= probabilities.sum() \n",
    "            self.item_choice = np.random.choice(len(self.values), p=probabilities)\n",
    "            if self.popularities[self.item_choice]<500:\n",
    "                self.item_rating = 1\n",
    "            else:\n",
    "                self.item_rating = 5\n",
    "        else:\n",
    "            raise ValueError(\"Invalid user type. Please choose a valid user type.\")\n",
    "        "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def update_user_tensor(user_vector, items, ratings):\n",
    "    Q_list = [item_vectors[item] for item in items]\n",
    "    Q = torch.tensor(Q_list, dtype=torch.float64)\n",
    "    # p = np.linalg.inv(Q.T @ Q) @ Q.T @ ratings\n",
    "    p = torch.inverse(Q.t() @ Q) @ Q.t() @ ratings\n",
    "    return p"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "file = open('items_rated', 'rb')\n",
    "\n",
    "# dump information to that file\n",
    "data = pickle.load(file)\n",
    "\n",
    "# close the file\n",
    "file.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "#5 is the user we are looking at\n",
    "chosen_item = data[5][-1][0]\n",
    "chosen_rating = data[5][-1][1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1258"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "chosen_item"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "file = open('ui_rating_dict', 'rb')\n",
    "\n",
    "# dump information to that file\n",
    "user_item_rating_dict = pickle.load(file)\n",
    "\n",
    "# close the file\n",
    "file.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "already_rated = list(user_item_rating_dict[5].keys())\n",
    "ratings = user_item_rating_dict[5]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/vibhhus/anaconda3/envs/py374/lib/python3.7/site-packages/ipykernel_launcher.py:2: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  \n"
     ]
    }
   ],
   "source": [
    "for i in ratings:\n",
    "    ratings[i]=torch.tensor(ratings[i], dtype=torch.float64)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "rating_tensor = torch.tensor(list(ratings.values()))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "ratings_dict = ratings"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/vibhhus/anaconda3/envs/py374/lib/python3.7/site-packages/ipykernel_launcher.py:32: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(-4.0901, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0237, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0938, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0831, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0888, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0858, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0916, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0895, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1049, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0953, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0990, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0417, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0435, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1039, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1058, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1076, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1100, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1215, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1118, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1117, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1136, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0596, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1173, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0632, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1207, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0668, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1264, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1260, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1320, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1319, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1321, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1473, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1389, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1377, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1396, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1447, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1433, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1439, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1484, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1621, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.0937, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1559, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1576, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1575, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1609, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1582, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1646, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1063, dtype=torch.float64, grad_fn=<NegBackward0>)\n",
      "tensor(-4.1656, dtype=torch.float64, grad_fn=<NegBackward0>)\n"
     ]
    }
   ],
   "source": [
    "#Stochastic choice\n",
    "user_action = torch.tensor(chosen_rating, requires_grad=True, dtype=torch.float64)\n",
    "optimizer = torch.optim.Adam([user_action], lr=0.05)\n",
    "item_to_be_reached = 6\n",
    "for epoch in range(1, 50):\n",
    "    user_action_clamped = user_action.clamp(1, 5) \n",
    "    # print(user_action_clamped)\n",
    "    # print(f\"User action: {user_action}\")\n",
    "    rating_tensor = torch.tensor(list(ratings_dict.values()))\n",
    "    user_vector_initial = torch.from_numpy(user_vectors[5])\n",
    "    user_vector = user_vector_initial\n",
    "    time_max = 1\n",
    "    already_rated = list(user_item_rating_dict[5].keys())\n",
    "    #print(already_rated)\n",
    "    #already_rated = already_rated[:-1]\n",
    "    ratings_old = rating_tensor#user_item_rating_dict[5]\n",
    "    # del ratings[chosen_item]\n",
    "    ratings_old[-1] = user_action#ratings[chosen_item] = user_action\n",
    "    #ratings_old_temp = torch.cat((ratings_old[:-1], user_action), dim=0)\n",
    "    n = len(ratings_old)\n",
    "    zeros_to_add = torch.zeros(time_max)\n",
    "    ratings = torch.cat((ratings_old, zeros_to_add), dim=0)\n",
    "    #print(ratings)\n",
    "    #already_rated.append(chosen_item)\n",
    "    user_vector = update_user_tensor(user_vector, already_rated, ratings[:n])\n",
    "    # for timestep in tqdm(range(1, time_max+1)):\n",
    "    for timestep in range(1, time_max+1):\n",
    "        recommendations, recommendation_scores = get_recommendation_scores(user_vector, already_rated)\n",
    "        #print(recommendation_scores)\n",
    "        usr_type = UserType(\"Choice Enjoyer\", recommendation_scores, [], beta=0.8)\n",
    "        choice_of_item = recommendations[usr_type.item_choice]\n",
    "        ratings[n+timestep-1] = torch.tensor(usr_type.item_rating)\n",
    "        already_rated.append(choice_of_item)\n",
    "        user_vector = update_user_tensor(user_vector, already_rated, ratings[:n+timestep])\n",
    "    item_rating = -torch.matmul(user_vector, torch.from_numpy(item_vectors[item_to_be_reached]))\n",
    "    print(item_rating)\n",
    "    #item_rating.requires_grad_()\n",
    "    #user_action.retain_grad()\n",
    "    # print(user_action.is_leaf)\n",
    "    item_rating.backward()\n",
    "    optimizer.step()\n",
    "    # print(\"Grad\")\n",
    "    # print(user_action.grad)\n",
    "    optimizer.zero_grad()\n",
    "    # print(user_action)\n",
    "    # \n",
    "    #user_action = user_action.clamp(1, 5) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Stochastic choice\n",
    "user_action = torch.tensor(chosen_rating, requires_grad=True, dtype=torch.float64)\n",
    "optimizer = torch.optim.SGD([user_action], lr=0.01)\n",
    "item_to_be_reached = 6\n",
    "for epoch in range(1, 50):\n",
    "    optimizer.zero_grad()\n",
    "    user_vector_initial = torch.from_numpy(user_vectors[5])\n",
    "    user_vector = user_vector_initial\n",
    "    time_max = 0\n",
    "    already_rated = list(user_item_rating_dict[5].keys())\n",
    "    #print(already_rated)\n",
    "    #already_rated = already_rated[:-1]\n",
    "    ratings_old = rating_tensor#user_item_rating_dict[5]\n",
    "    # del ratings[chosen_item]\n",
    "    ratings_old[-1] = user_action#ratings[chosen_item] = user_action\n",
    "    #ratings_old_temp = torch.cat((ratings_old[:-1], user_action), dim=0)\n",
    "    n = len(ratings_old)\n",
    "    zeros_to_add = torch.zeros(time_max)\n",
    "    ratings = torch.cat((ratings_old, zeros_to_add), dim=0)\n",
    "    #print(ratings)\n",
    "    #already_rated.append(chosen_item)\n",
    "    user_vector = update_user_tensor(user_vector, already_rated, ratings[:n])\n",
    "    for timestep in tqdm(range(1, time_max+1)):\n",
    "        recommendations, recommendation_scores = get_recommendation_scores(user_vector, already_rated)\n",
    "        print(recommendation_scores)\n",
    "        usr_type = UserType(\"Choice Enjoyer\", recommendation_scores, [], beta=0.8)\n",
    "        choice_of_item = recommendations[usr_type.item_choice]\n",
    "        ratings[n+timestep-1] = torch.tensor(usr_type.item_rating)\n",
    "        already_rated.append(choice_of_item)\n",
    "        user_vector = update_user_tensor(user_vector, already_rated, ratings[:n+timestep])\n",
    "    item_rating = -torch.matmul(user_vector, torch.from_numpy(item_vectors[item_to_be_reached]))\n",
    "    print(item_rating)\n",
    "    #item_rating.requires_grad_()\n",
    "    #user_action.retain_grad()\n",
    "    print(user_action.is_leaf)\n",
    "    item_rating.backward()\n",
    "    optimizer.step()\n",
    "    print(user_action.grad)\n",
    "    # print(user_action)\n",
    "    user_action = torch.clamp(user_action, min=1, max=5)\n",
    "    #user_action = user_action.clamp(1, 5) \n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "py374",
   "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.7.4"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
