{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "dfa2c759",
   "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",
    "\n",
    "from dgl.data import RedditDataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "bdb10d2a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  NumNodes: 2708\n",
      "  NumEdges: 10556\n",
      "  NumFeats: 1433\n",
      "  NumClasses: 7\n",
      "  NumTrainingSamples: 140\n",
      "  NumValidationSamples: 500\n",
      "  NumTestSamples: 1000\n",
      "Done loading data from cached files.\n"
     ]
    }
   ],
   "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",
    "graph, feat, labels, train_mask, val_mask, test_mask, number_classes = load_graph_dataset(dataname)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "2a41a30e",
   "metadata": {},
   "outputs": [],
   "source": [
    "lr = SimplifiedGraphNeuralNetwork(l2_reg=l2_regularlization_term, fit_intercept=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "7c95a191",
   "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": 5,
   "id": "35302f31",
   "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": 6,
   "id": "5bed77d6",
   "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[test_mask].numpy().astype(np.float32)\n",
    "test_y = labels[test_mask].numpy().astype(np.float32)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "f1835314",
   "metadata": {},
   "outputs": [],
   "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",
    "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))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "6777d084",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 139/139 [00:00<00:00, 210.63it/s]\n"
     ]
    }
   ],
   "source": [
    "train_total_grad, train_indiv_grad = lr.grad(train_x, logits_train_y, \n",
    "                                             one_hot_labels_train, l2_reg=True)\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": 9,
   "id": "4638ccc2",
   "metadata": {},
   "outputs": [],
   "source": [
    "_, a = lr.grad_one_batch(train_x, logits_train_y, one_hot_labels_train, l2_reg=True, batch_index=np.array([1, 2]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "57a57fe5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[ 0.        ,  0.        ,  0.        , ..., -0.62935859,\n",
       "         0.16057255,  0.15657205],\n",
       "       [ 0.        ,  0.        ,  0.        , ...,  0.04241721,\n",
       "        -0.35342934,  0.0735105 ]])"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "a"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "46e67223",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "30a7d9c2",
   "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": 12,
   "id": "38606b36",
   "metadata": {},
   "outputs": [],
   "source": [
    "for i in range(num_train):\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": 13,
   "id": "598e399b",
   "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": null,
   "id": "5e5a4227",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "1cdd24e7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "SpearmanrResult(correlation=0.9781198749207323, pvalue=5.566055357789712e-96)"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import scipy\n",
    "scipy.stats.spearmanr(act_infl, pred_infl)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "cab07d57",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1117.3319167091167"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ori_val_loss"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "ed760540",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.DataFrame([pred_infl, act_infl]).T\n",
    "df.columns = ['predicted influence', 'actual influence']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "bae55efc",
   "metadata": {},
   "outputs": [],
   "source": [
    "# df.to_csv('node_feature/' + dataname +'.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "f8b61f08",
   "metadata": {},
   "outputs": [],
   "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
}
