{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib\n",
    "from matplotlib import pyplot as plt\n",
    "import os\n",
    "from sklearn.metrics import mean_squared_error as mse\n",
    "from scipy import sparse\n",
    "from scipy.stats import gamma"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_set_overlap(Beta_p, Beta, k=50):\n",
    "    top = np.argsort(Beta)[-k:]\n",
    "    top_p = np.argsort(Beta_p)[-k:]\n",
    "    return np.intersect1d(top, top_p).shape[0]/np.union1d(top, top_p).shape[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "sim_name = 'sim_inf_1k_users'\n",
    "fitted_params = {}\n",
    "true_params = {}\n",
    "random_params = {}\n",
    "spf_params = {}\n",
    "adjusted_params = {}\n",
    "inf_only_params = {}\n",
    "\n",
    "alphas = [0.01, 0.05, 0.1, 0.5, 1.0]\n",
    "datadir = os.path.join('..','dat',sim_name)\n",
    "for alpha in alphas:\n",
    "    fitted_beta = np.loadtxt(datadir + '/out/simulation_fitted_params/alpha=' + str(alpha) + '_fitted.gz')\n",
    "    fitted_params[alpha] = fitted_beta\n",
    "    \n",
    "    true_beta = np.loadtxt(datadir + '/alpha=' + str(alpha) + '/influence.gz')\n",
    "    true_params[alpha] = true_beta\n",
    "    \n",
    "    N = len(true_beta)\n",
    "    random_params[alpha] = gamma.rvs(alpha, scale=10., size=N)\n",
    "    \n",
    "    spf_fitted_beta = np.loadtxt(datadir + '/out/spf_fitted_params/alpha=' + str(alpha) + '_fitted.gz')\n",
    "    spf_params[alpha] = spf_fitted_beta\n",
    "    \n",
    "    adjusted_fitted_beta = np.loadtxt(datadir + '/out/adjusted_model_fitted_params/alpha=' + str(alpha) + '_fitted.gz')\n",
    "    adjusted_params[alpha] = adjusted_fitted_beta"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Alpha: 0.01 **********\n",
      "Random RMSE: 0.8387956310979651\n",
      "Random Overlap: 0.02040816326530612\n",
      "\n",
      "\n",
      "SPF RMSE: 2.0675980071640057\n",
      "SPF Overlap: 0.041666666666666664\n",
      "\n",
      "\n",
      "Adjusted RMSE: 0.36209340085416586\n",
      "Adjusted Overlap: 0.28205128205128205\n",
      "\n",
      "\n",
      "RMSE: 0.3641445211956709\n",
      "Overlap: 0.3157894736842105\n",
      "--------------------\n",
      "Alpha: 0.05 **********\n",
      "Random RMSE: 2.8845308268340566\n",
      "Random Overlap: 0.010101010101010102\n",
      "\n",
      "\n",
      "SPF RMSE: 2.152519775133319\n",
      "SPF Overlap: 0.21951219512195122\n",
      "\n",
      "\n",
      "Adjusted RMSE: 0.8327456881318432\n",
      "Adjusted Overlap: 0.7543859649122807\n",
      "\n",
      "\n",
      "RMSE: 0.8017686798149017\n",
      "Overlap: 0.8181818181818182\n",
      "--------------------\n",
      "Alpha: 0.1 **********\n",
      "Random RMSE: 5.368036728061969\n",
      "Random Overlap: 0.010101010101010102\n",
      "\n",
      "\n",
      "SPF RMSE: 2.524785299020917\n",
      "SPF Overlap: 0.3698630136986301\n",
      "\n",
      "\n",
      "Adjusted RMSE: 1.8795365029162052\n",
      "Adjusted Overlap: 0.639344262295082\n",
      "\n",
      "\n",
      "RMSE: 1.0537030712264244\n",
      "Overlap: 0.8181818181818182\n",
      "--------------------\n",
      "Alpha: 0.5 **********\n",
      "Random RMSE: 9.772647458606103\n",
      "Random Overlap: 0.010101010101010102\n",
      "\n",
      "\n",
      "SPF RMSE: 3.857071409538288\n",
      "SPF Overlap: 0.639344262295082\n",
      "\n",
      "\n",
      "Adjusted RMSE: 4.0158157307750235\n",
      "Adjusted Overlap: 0.6666666666666666\n",
      "\n",
      "\n",
      "RMSE: 3.1271185914923314\n",
      "Overlap: 0.7857142857142857\n",
      "--------------------\n",
      "Alpha: 1.0 **********\n",
      "Random RMSE: 13.825847229504777\n",
      "Random Overlap: 0.010101010101010102\n",
      "\n",
      "\n",
      "SPF RMSE: 4.641834109174613\n",
      "SPF Overlap: 0.6949152542372882\n",
      "\n",
      "\n",
      "Adjusted RMSE: 6.0368934985624305\n",
      "Adjusted Overlap: 0.639344262295082\n",
      "\n",
      "\n",
      "RMSE: 4.709549667968947\n",
      "Overlap: 0.7857142857142857\n",
      "--------------------\n"
     ]
    }
   ],
   "source": [
    "k=50\n",
    "for a in alphas:\n",
    "    print(\"Alpha:\", a, \"*\"*10)\n",
    "    fitted = fitted_params[a]\n",
    "    truth = true_params[a]\n",
    "    random = random_params[a]\n",
    "    spf = spf_params[a]\n",
    "    adjusted = adjusted_params[a]\n",
    "    \n",
    "    print(\"Random RMSE:\", np.sqrt(mse(truth, random)))\n",
    "    print(\"Random Overlap:\", get_set_overlap(random, truth, k=k))\n",
    "    print('\\n')\n",
    "    \n",
    "    print(\"SPF RMSE:\", np.sqrt(mse(truth, spf)))\n",
    "    print(\"SPF Overlap:\", get_set_overlap(spf, truth, k=k))\n",
    "    print('\\n')\n",
    "    \n",
    "    print(\"Adjusted RMSE:\", np.sqrt(mse(truth, adjusted)))\n",
    "    print(\"Adjusted Overlap:\", get_set_overlap(adjusted, truth, k=k))\n",
    "    print('\\n')\n",
    "    \n",
    "    print(\"RMSE:\", np.sqrt(mse(truth, fitted)))\n",
    "    print(\"Overlap:\", get_set_overlap(fitted, truth, k=k))\n",
    "    print(\"-\"*20)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Compute a very naive ranking of influential users"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Naive influence: 0.08695652173913043\n",
      "Naive influence: 0.12359550561797752\n",
      "Naive influence: 0.06382978723404255\n",
      "Naive influence: 0.06382978723404255\n",
      "Naive influence: 0.041666666666666664\n"
     ]
    }
   ],
   "source": [
    "Y_p = sparse.load_npz(datadir + '/alpha=1.0/past_obs.gz.npz').toarray()\n",
    "A = sparse.load_npz(datadir + '/alpha=1.0/adj.gz.npz')\n",
    "alphas = [0.01, 0.05, 0.1, 0.5, 1.0]\n",
    "for alpha in alphas:\n",
    "    Y = sparse.load_npz(datadir + '/alpha=' + str(alpha) + '/obs.gz.npz')\n",
    "    friend_items = A.T.dot(Y)\n",
    "    item_overlap = Y_p*friend_items.toarray()\n",
    "    influence_measure = item_overlap.sum(axis=1)\n",
    "    overlap = get_set_overlap(influence_measure, true_params[alpha])\n",
    "    print(\"Naive influence:\",overlap)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Matching algorithm (similar to Sharma et al. 2016) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[0.        , 0.        , 0.        , ..., 0.01652893, 0.00961538,\n",
       "        0.        ],\n",
       "       [0.        , 0.        , 0.        , ..., 0.        , 0.03631285,\n",
       "        0.        ],\n",
       "       [0.        , 0.        , 0.        , ..., 0.        , 0.        ,\n",
       "        0.        ],\n",
       "       ...,\n",
       "       [0.01652893, 0.        , 0.        , ..., 0.        , 0.        ,\n",
       "        0.        ],\n",
       "       [0.00961538, 0.03631285, 0.        , ..., 0.        , 0.        ,\n",
       "        0.        ],\n",
       "       [0.        , 0.        , 0.        , ..., 0.        , 0.        ,\n",
       "        0.        ]])"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from itertools import combinations\n",
    "N = Y_p.shape[0]\n",
    "similarity = np.zeros((N, N))\n",
    "users = np.arange(N)\n",
    "for (i,j) in combinations(users, 2):\n",
    "    items_i = np.nonzero(Y_p[i,:])[0]\n",
    "    items_j = np.nonzero(Y_p[j,:])[0]\n",
    "\n",
    "    if items_i.shape[0] >0 and items_j.shape[0] > 0:\n",
    "        overlap = np.intersect1d(items_i, items_j).shape[0]/np.union1d(items_i, items_j).shape[0]\n",
    "    else:\n",
    "        overlap=0.0\n",
    "    similarity[i][j] = overlap\n",
    "    similarity[j][i] = overlap"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "A = A.toarray()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.1111111111111111\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-22-65aa251f8fe8>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      9\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0mv\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mneighbors\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     10\u001b[0m             \u001b[0mu_embed\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0muser_embed\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mu\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 11\u001b[0;31m             \u001b[0msimilarity\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mlinalg\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mnorm\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mu_embed\u001b[0m\u001b[0;34m-\u001b[0m\u001b[0muser_embed\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0md\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mfor\u001b[0m \u001b[0md\u001b[0m \u001b[0;32min\u001b[0m \u001b[0musers\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     12\u001b[0m             \u001b[0mmatch\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m(\u001b[0m\u001b[0msimilarity\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m-\u001b[0m\u001b[0mA\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mv\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0margmin\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     13\u001b[0m             \u001b[0mitems_v\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mnonzero\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mY\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mv\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m<ipython-input-22-65aa251f8fe8>\u001b[0m in \u001b[0;36m<listcomp>\u001b[0;34m(.0)\u001b[0m\n\u001b[1;32m      9\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0mv\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mneighbors\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     10\u001b[0m             \u001b[0mu_embed\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0muser_embed\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mu\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 11\u001b[0;31m             \u001b[0msimilarity\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mlinalg\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mnorm\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mu_embed\u001b[0m\u001b[0;34m-\u001b[0m\u001b[0muser_embed\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0md\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mfor\u001b[0m \u001b[0md\u001b[0m \u001b[0;32min\u001b[0m \u001b[0musers\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     12\u001b[0m             \u001b[0mmatch\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m(\u001b[0m\u001b[0msimilarity\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m-\u001b[0m\u001b[0mA\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mv\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0margmin\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     13\u001b[0m             \u001b[0mitems_v\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mnonzero\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mY\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mv\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "user_embed = np.loadtxt(datadir + '/alpha=0.1/user_embed.gz')\n",
    "for alpha in alphas:\n",
    "    Y = sparse.load_npz(datadir + '/alpha=' + str(alpha) + '/obs.gz.npz').toarray()\n",
    "    influence = np.zeros(N)\n",
    "    for u in users:\n",
    "        neighbors = np.nonzero(A[:, u])[0]\n",
    "        size = neighbors.shape[0]\n",
    "        items_u = np.nonzero(Y_p[u,:])[0]\n",
    "        for v in neighbors:\n",
    "#             u_embed = user_embed[u, :]\n",
    "#             similarity = [np.linalg.norm(u_embed-user_embed[d,:]) for d in users]\n",
    "            match = (similarity[u,:]*(1-A[v,:])).argmin()\n",
    "            items_v = np.nonzero(Y[v,:])[0]\n",
    "            items_match = np.nonzero(Y_p[match,:])[0]\n",
    "            match_overlap = np.intersect1d(items_v, items_match).shape[0]\n",
    "            inf_overlap = np.intersect1d(items_v, items_u).shape[0]\n",
    "            if items_v.shape[0] == 0:\n",
    "                inf = 0.0\n",
    "            else:\n",
    "                inf = (inf_overlap - match_overlap)/items_v.shape[0]\n",
    "            influence[u]+= np.abs(inf)\n",
    "        influence[u] /= size\n",
    "    print(get_set_overlap(influence, true_params[alpha]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(overlaps, np.arange(0.1, 1.1, step=0.1))\n",
    "plt.ylabel('Alpha values')\n",
    "plt.xlabel('Jaccard overlap in top 100 influential users')\n",
    "plt.savefig('../doc/lastfm-inf-exp.png')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "user_embed = np.loadtxt(datadir + '/alpha=0.1/user_embed.gz')\n",
    "Y_p = sparse.load_npz(datadir + '/alpha=0.1/past_obs.gz.npz')\n",
    "Y_p = Y_p.toarray()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_users = user_embed.shape[0]\n",
    "matched_pairs = []\n",
    "for u in range(num_users):\n",
    "    u_embed = user_embed[u, :]\n",
    "    dists = [np.linalg.norm(u_embed-user_embed[d,:]) for d in range(num_users) if d!=u]\n",
    "    match = np.argsort(dists)[0]\n",
    "    matched_pairs.append((u,match))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.011445796106166976"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "overlaps = []\n",
    "for (i,j) in matched_pairs:\n",
    "    items_i = np.nonzero(Y_p[i,:])[0]\n",
    "    items_j = np.nonzero(Y_p[j,:])[0]\n",
    "\n",
    "    if items_i.shape[0] >0 and items_j.shape[0] > 0:\n",
    "        overlap = np.intersect1d(items_i, items_j).shape[0]/np.union1d(items_i, items_j).shape[0]\n",
    "    else:\n",
    "        overlap=0.0\n",
    "    overlaps.append(overlap)\n",
    "np.mean(overlaps)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.022061448221832973"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "A = sparse.load_npz(datadir + '/alpha=0.1/adj.gz.npz')\n",
    "A = A.toarray()\n",
    "overlaps = []\n",
    "for u in range(num_users):\n",
    "    for j in np.nonzero(A[u,:])[0]:\n",
    "        items_i = np.nonzero(Y_p[u,:])[0]\n",
    "        items_j = np.nonzero(Y_p[j,:])[0]\n",
    "\n",
    "        if items_i.shape[0] >0 and items_j.shape[0] > 0:\n",
    "            overlap = np.intersect1d(items_i, items_j).shape[0]/np.union1d(items_i, items_j).shape[0]\n",
    "        else:\n",
    "            overlap=0.0\n",
    "        overlaps.append(overlap)\n",
    "        \n",
    "np.mean(overlaps)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.6.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
