{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "0000823d",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Using backend: pytorch\n",
      "Intel(R) Extension for Scikit-learn* enabled (https://github.com/intel/scikit-learn-intelex)\n"
     ]
    }
   ],
   "source": [
    "import sys\n",
    "sys.path.append('..')\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import torch\n",
    "import os\n",
    "import dgl\n",
    "from dgl import function as fn\n",
    "from dataset import load_graph_dataset\n",
    "from model_softmax import SimplifiedGraphNeuralNetwork, fast_hess, fast_hess_cuda, fast_get_inv_hvp_cuda\n",
    "import tensorflow.compat.v1 as tf\n",
    "from graph_neural_networks import SGC_layer1, SGC_layer2\n",
    "from sklearn.preprocessing import OneHotEncoder\n",
    "from sklearn.metrics import log_loss\n",
    "from scipy.special import softmax, log_softmax\n",
    "from scipy.linalg import cho_solve, cho_factor\n",
    "from tqdm import tqdm\n",
    "import cupy as cp\n",
    "import time\n",
    "from dgl.data import RedditDataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "07d18aa8",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Intel(R) Extension for Scikit-learn* enabled (https://github.com/intel/scikit-learn-intelex)\n"
     ]
    }
   ],
   "source": [
    "from sklearnex import patch_sklearn, config_context\n",
    "patch_sklearn()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "c0683743",
   "metadata": {},
   "outputs": [],
   "source": [
    "# dataname = 'pubmed'\n",
    "# l2_regularlization_term = 0.004\n",
    "\n",
    "# dataname = 'cora'\n",
    "# l2_regularlization_term = 0.01\n",
    "\n",
    "# dataname = 'pubmed'\n",
    "# l2_regularlization_term = 0.004\n",
    "# num_layer = 2\n",
    "\n",
    "# dataname = 'citeseer'\n",
    "# l2_regularlization_term = 0.003\n",
    "# num_layer = 2\n",
    "\n",
    "\n",
    "dataname = 'reddit'\n",
    "l2_regularlization_term = 0.99\n",
    "num_layer = 2\n",
    "\n",
    "graph, feat, labels, train_mask, val_mask, test_mask, number_classes = load_graph_dataset(dataname)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "894343f5",
   "metadata": {},
   "outputs": [],
   "source": [
    "# lr = SimplifiedGraphNeuralNetwork(l2_reg=l2_regularlization_term, fit_intercept=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "50d2cefa",
   "metadata": {},
   "outputs": [],
   "source": [
    "# from generate_mid_layer_feature import FeatureExtraction\n",
    "\n",
    "# FeatureExtractor = FeatureExtraction(num_layers=2, num_iter=100, lr=0.02, hidden_feat=20, device='cpu',\n",
    "#                                          dataset='pubmed')\n",
    "# FeatureExtractor.extract_feature()\n",
    "# train_x, train_y, val_x, val_y, test_x, test_y = FeatureExtractor.preprocessed_data(save='feat.csv')\n",
    "\n",
    "\n",
    "# train_x = train_x.astype(np.float64)\n",
    "# test_x = test_x.astype(np.float64)\n",
    "# train_y = train_y.astype(np.float64)\n",
    "# test_y = test_y.astype(np.float64)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "e7b0b24e",
   "metadata": {},
   "outputs": [],
   "source": [
    "feat0 = feat.clone()\n",
    "degs = graph.in_degrees().float().clamp(min = 1)\n",
    "norm = torch.pow(degs, -0.5)\n",
    "norm = norm.to(feat0.device).unsqueeze(1)\n",
    "\n",
    "for _ in range(2):\n",
    "    feat0 = feat0 * norm\n",
    "    graph.ndata['h'] = feat0\n",
    "    graph.update_all(fn.copy_u('h', 'm'),\n",
    "                     fn.sum('m', 'h'))\n",
    "    feat0 = graph.ndata.pop('h')\n",
    "    feat0 = feat0 * norm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "8ea2b434",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_x = feat0[train_mask].numpy().astype(np.float32)\n",
    "train_y = labels[train_mask].numpy().astype(np.float32)\n",
    "\n",
    "test_x = feat0[val_mask].numpy().astype(np.float32)\n",
    "test_y = labels[val_mask].numpy().astype(np.float32)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "deb6842e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--- 84.26344037055969 seconds ---\n"
     ]
    }
   ],
   "source": [
    "enc = OneHotEncoder(handle_unknown='ignore')\n",
    "enc.fit(train_y.reshape(-1, 1))\n",
    "\n",
    "one_hot_labels_train = enc.transform(train_y.reshape(-1, 1)).toarray()\n",
    "one_hot_labels_test = enc.transform(test_y.reshape(-1, 1)).toarray()\n",
    "\n",
    "\"\"\" Train Logistic Regression \"\"\"\n",
    "\n",
    "import time\n",
    "start_time = time.time()\n",
    "\n",
    "lr = SimplifiedGraphNeuralNetwork(l2_reg = l2_regularlization_term, fit_intercept=True)\n",
    "lr.fit(train_x, train_y, sample_weight=None, verbose=False)\n",
    "# logits_test_y = test_x @ lr.model.coef_.T + lr.model.intercept_\n",
    "# logits_train_y = train_x @ lr.model.coef_.T + lr.model.intercept_\n",
    "\n",
    "# ori_val_loss, ave_ori_val_loss = lr.log_loss(logits_test_y, one_hot_labels_test, l2_reg = True)\n",
    "\n",
    "# numpy_theoritic_loss = log_loss(test_y, softmax(logits_test_y, axis=1))\n",
    "\n",
    "\n",
    "print(\"--- %s seconds ---\" % (time.time() - start_time))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "d9d5fe58",
   "metadata": {},
   "outputs": [],
   "source": [
    "logits_train_y = train_x @ lr.model.coef_.T + lr.model.intercept_"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "de08add8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "153431"
      ]
     },
     "execution_count": 48,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train_x.shape[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "a97b9320",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "4751063e",
   "metadata": {},
   "outputs": [],
   "source": [
    "# num_train = train_x.shape[0]\n",
    "# batch = 10000\n",
    "# idx_all = np.arange(0, num_train)\n",
    "# for i in tqdm(range(0, num_train, batch)):\n",
    "# #     print(idx_all[i: i + batch])\n",
    "#     current_batch_index = idx_all[i:i+batch]\n",
    "#     _, temp_train_indiv_grad = lr.grad_one_batch(train_x, logits_train_y, \n",
    "#                                                  one_hot_labels_train, l2_reg=True, \n",
    "#                                                  fit_intercept=True, batch_index=current_batch_index)\n",
    "#     pd.DataFrame(temp_train_indiv_grad).to_csv('reddit/training_gradients/'+str(i) + '_' + \n",
    "#                                                str(i+ batch) + '.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "1d260ee3",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "46af3d9a",
   "metadata": {},
   "outputs": [],
   "source": [
    "\"\"\"\n",
    "train_total_grad, train_indiv_grad = lr.grad(train_x, logits_train_y, \n",
    "                                             one_hot_labels_train, l2_reg=True)\n",
    "\n",
    "\n",
    "\"\"\"\n",
    "\n",
    "# val_loss_total_grad, val_loss_indiv_grad = lr.grad(test_x, logits_test_y, \n",
    "#                                                    one_hot_labels_test, l2_reg=True)\n",
    "\n",
    "# hessian_no_reg, hess, hessian_reg_term = lr.hess(train_x, logits_train_y)\n",
    "# hess = fast_hess_cuda(train_x, logits_train_y)\n",
    "# hess = lr.hess_cuda(train_x, logits_train_y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a4ae3188",
   "metadata": {},
   "outputs": [],
   "source": [
    "# hess = lr.hess_cuda(train_x, logits_train_y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 84,
   "id": "c00801cf",
   "metadata": {},
   "outputs": [],
   "source": [
    "loss_grad_hvp = np.array(pd.read_csv('reddit/loss_grad_hvp.csv'))\n",
    "ori_val_loss = np.array(pd.read_csv('reddit/ori_val_loss.csv'))[0][0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "bfbd59af",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# loss_grad_hvp = fast_get_inv_hvp_cuda(hess, val_loss_total_grad.T, cholskey=True)\n",
    "# # loss_grad_hvp = fast_get_inv_hvp_cuda(hess, val_loss_total_grad.T, cholskey=False)\n",
    "# loss_grad_hvp = cp.asnumpy(loss_grad_hvp)\n",
    "\n",
    "# pred_infl = train_indiv_grad.dot(loss_grad_hvp)\n",
    "\n",
    "# pred_infl = list(pred_infl.reshape(-1))\n",
    "# #\n",
    "# num_train = len(train_x)\n",
    "# act_infl = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "id": "cc219127",
   "metadata": {},
   "outputs": [],
   "source": [
    "import random\n",
    "random.seed(42)\n",
    "num_rand = 10000\n",
    "rand_idx = random.sample(range(0, num_rand), 100)\n",
    "rand_idx = np.sort(rand_idx)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "id": "9ae6dda3",
   "metadata": {},
   "outputs": [],
   "source": [
    "_, temp_train_indiv_grad = lr.grad_one_batch(train_x, logits_train_y, \n",
    "                                                 one_hot_labels_train, l2_reg=True, \n",
    "                                                 fit_intercept=True, batch_index=rand_idx)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "id": "25702a8b",
   "metadata": {},
   "outputs": [],
   "source": [
    "pred_infl = temp_train_indiv_grad.dot(loss_grad_hvp)\n",
    "\n",
    "pred_infl = list(pred_infl.reshape(-1))\n",
    "#\n",
    "num_train = len(train_x)\n",
    "act_infl = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 88,
   "id": "2a0c2748",
   "metadata": {},
   "outputs": [],
   "source": [
    "# pd.DataFrame(rand_idx).to_csv('reddit/rand_idx.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "id": "938f7be5",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 100/100 [2:13:19<00:00, 79.99s/it]\n"
     ]
    }
   ],
   "source": [
    "# for i in range(num_train):\n",
    "for i in tqdm(rand_idx):\n",
    "    lr_new = SimplifiedGraphNeuralNetwork(l2_reg=l2_regularlization_term, fit_intercept=True)\n",
    "    train_x_new = np.delete(train_x, i, axis = 0)\n",
    "    train_y_new = np.delete(train_y, i)\n",
    "    lr_new.fit(train_x_new, train_y_new)\n",
    "    \n",
    "    logits_test_y_new = test_x @ lr_new.model.coef_.T + lr_new.model.intercept_\n",
    "    \n",
    "    \n",
    "    new_ori_val_loss, new_ave_ori_val_loss = lr_new.log_loss(logits_test_y_new, one_hot_labels_test, l2_reg = True)\n",
    "    act_infl.append(new_ori_val_loss - ori_val_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "id": "dd2f6e0c",
   "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": [
    "import matplotlib.pyplot as plt\n",
    "x = np.linspace(-10, 10)\n",
    "# x = np.linspace(-0.2, 0.15)\n",
    "plt.plot(x, x)\n",
    "plt.plot(act_infl, pred_infl, 'o', color='blue')\n",
    "plt.xlabel('actual change in loss')\n",
    "plt.ylabel('predicted change in loss')\n",
    "# plt.title('Influence function on Cora dataset')\n",
    "plt.title('Influence function on Citeseer dataset - Node Feature')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 94,
   "id": "73221684",
   "metadata": {},
   "outputs": [],
   "source": [
    "pd.DataFrame(np.array([act_infl, pred_infl]).T).to_csv('reddit/reddit_node_feature_influence.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "1c68f670",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "SpearmanrResult(correlation=0.9757673723317927, pvalue=1.0083090441525486e-79)"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import scipy\n",
    "scipy.stats.spearmanr(act_infl, pred_infl)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "b2176ed0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1114.1484784469749"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ori_val_loss"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "7d6b7dcb",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.DataFrame([pred_infl, act_infl]).T\n",
    "df.columns = ['predicted influence', 'actual influence']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "ff0421d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "df.to_csv('node_feature/' + dataname +'.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "92af961a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>predicted influence</th>\n",
       "      <th>actual influence</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>3.299618</td>\n",
       "      <td>5.340206</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>3.190782</td>\n",
       "      <td>4.843376</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>0.289123</td>\n",
       "      <td>1.269817</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>0.466144</td>\n",
       "      <td>0.729692</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>0.452809</td>\n",
       "      <td>1.781227</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>135</th>\n",
       "      <td>2.537044</td>\n",
       "      <td>3.645985</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>136</th>\n",
       "      <td>-1.516677</td>\n",
       "      <td>-1.688545</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>137</th>\n",
       "      <td>3.535811</td>\n",
       "      <td>4.865627</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>138</th>\n",
       "      <td>3.004675</td>\n",
       "      <td>5.169648</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>139</th>\n",
       "      <td>3.427027</td>\n",
       "      <td>5.038287</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>140 rows × 2 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "     predicted influence  actual influence\n",
       "0               3.299618          5.340206\n",
       "1               3.190782          4.843376\n",
       "2               0.289123          1.269817\n",
       "3               0.466144          0.729692\n",
       "4               0.452809          1.781227\n",
       "..                   ...               ...\n",
       "135             2.537044          3.645985\n",
       "136            -1.516677         -1.688545\n",
       "137             3.535811          4.865627\n",
       "138             3.004675          5.169648\n",
       "139             3.427027          5.038287\n",
       "\n",
       "[140 rows x 2 columns]"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# df = pd.read_csv('hyper_parameter/cora.csv')\n",
    "pd.read_csv('node_feature/cora.csv')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.8.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
