{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "95b852b9",
   "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",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import os\n",
    "import math\n",
    "import torch\n",
    "import dgl\n",
    "import numpy_ml\n",
    "import torch.nn.functional as F\n",
    "import tensorflow.compat.v1 as tf\n",
    "import scipy\n",
    "import cupy as cp\n",
    "import cupyx\n",
    "\n",
    "from dgl import function as fn\n",
    "from dgl.base import DGLError\n",
    "from generate_mid_layer_feature import FeatureExtraction\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "from scipy.special import softmax, log_softmax\n",
    "from numpy.linalg import inv, pinv\n",
    "from sklearn.preprocessing import OneHotEncoder\n",
    "from sklearn.metrics import log_loss\n",
    "from sklearn.preprocessing import LabelBinarizer\n",
    "from sklearn.utils import check_array\n",
    "from dataset import load_graph_dataset\n",
    "from sklearn.datasets import load_iris\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "from model_node_influence import NodeInfluenceSGC\n",
    "\n",
    "from model_softmax import SimplifiedGraphNeuralNetwork, fast_hess, fast_hess_cuda, fast_get_inv_hvp_cuda\n",
    "\n",
    "from matplotlib import pyplot as plt\n",
    "from scipy.linalg import cho_solve, cho_factor\n",
    "from model_edge_influence import EdgeInfluenceSGC, generate_remove_index_train\n",
    "from tqdm import tqdm\n",
    "import networkx as nx"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "4738c4db",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'/home/zizhang/Desktop/Projects/Project6_influence_function/graph_influence_function/experiments'"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "os.getcwd()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "9a7ca915",
   "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": 4,
   "id": "773fafe9",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_set = 'cora'\n",
    "# l2_term = 0.01\n",
    "l2_term = 0.01\n",
    "# batch_edges = 1\n",
    "num_layer = 2\n",
    "\n",
    "# data_set = 'pubmed'\n",
    "# l2_term = 0.004\n",
    "# num_layer = 2\n",
    "\n",
    "# data_set = 'citeseer'\n",
    "# l2_term = 0.003\n",
    "# num_layer = 2\n",
    "\n",
    "# data_set = 'reddit'\n",
    "# l2_term = 0.99\n",
    "# num_layer = 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "611b0bbe",
   "metadata": {},
   "outputs": [],
   "source": [
    "graph, feat, labels, train_mask, val_mask, test_mask, number_classes = load_graph_dataset(data_set)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "52128482",
   "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(num_layer):\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": "ec0bf1bb",
   "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)\n",
    "\n",
    "val_x = feat0[val_mask].numpy().astype(np.float32)\n",
    "val_y = labels[val_mask].numpy().astype(np.float32)\n",
    "\n",
    "train_node_idx = torch.where(train_mask == 1)[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "d91efd4b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "' Train Logistic Regression '"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "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",
    "one_hot_labels_val = enc.transform(val_y.reshape(-1, 1)).toarray()\n",
    "\"\"\" Train Logistic Regression \"\"\"\n",
    "# lr = SimplifiedGraphNeuralNetwork(l2_reg=l2_term, fit_intercept=True)\n",
    "# lr.fit(train_x, train_y, sample_weight=None, verbose=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "d66ddb88",
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_remove_index_train_all(from_indexes, to_indexes, train_mask, seed_val=10):\n",
    "    torch.manual_seed(seed_val)\n",
    "    train_index = torch.where(train_mask == 1)[0]\n",
    "    remove_from_list = []\n",
    "    remove_to_list = []\n",
    "    for i in tqdm(range(len(train_index))):\n",
    "        f_index = train_index[i]\n",
    "\n",
    "        to_index_list = torch.where(from_indexes == f_index)[0]\n",
    "        \n",
    "#         print(f_index)\n",
    "#         random_index = torch.randint(0, len(to_index_list), (1,))[0]\n",
    "        for to_index_e in to_index_list:\n",
    "#             print(to_index_list)\n",
    "#             print(to_index_e.item())\n",
    "#             j = to_index_list[to_index_e[0]]\n",
    "            j = to_index_e\n",
    "            t_index = to_indexes[j]\n",
    "\n",
    "            remove_from_list.append(f_index)\n",
    "            remove_to_list.append(t_index)\n",
    "\n",
    "    return torch.tensor(remove_from_list), torch.tensor(remove_to_list)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "d2871668",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|▏                                   | 691/153431 [00:41<2:31:38, 16.79it/s]\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/tmp/ipykernel_5796/166491003.py\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      1\u001b[0m \u001b[0mfrom_indexes\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mto_indexes\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mgraph\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0medges\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[1;32m      2\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 3\u001b[0;31m \u001b[0mf_l\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mt_l\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mgenerate_remove_index_train_all\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfrom_indexes\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mto_indexes\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtrain_mask\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m      4\u001b[0m \u001b[0mf_l\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mt_l\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mfrom_indexes\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mto_indexes\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      5\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/tmp/ipykernel_5796/181964420.py\u001b[0m in \u001b[0;36mgenerate_remove_index_train_all\u001b[0;34m(from_indexes, to_indexes, train_mask, seed_val)\u001b[0m\n\u001b[1;32m      7\u001b[0m         \u001b[0mf_index\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtrain_index\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mi\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      8\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 9\u001b[0;31m         \u001b[0mto_index_list\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mwhere\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfrom_indexes\u001b[0m \u001b[0;34m==\u001b[0m \u001b[0mf_index\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[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     10\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     11\u001b[0m \u001b[0;31m#         print(f_index)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "from_indexes, to_indexes = graph.edges()\n",
    "\n",
    "f_l, t_l = generate_remove_index_train_all(from_indexes, to_indexes, train_mask)\n",
    "f_l, t_l = from_indexes, to_indexes\n",
    "\n",
    "# acctual_influence_1 = []\n",
    "# acctual_influence_2 = []\n",
    "\n",
    "# predict_influence_1 = []\n",
    "# predict_influence_2 = []\n",
    "\n",
    "# df = pd.read_csv('result_data/' + data_set + '_edge_influence.csv', header = None)\n",
    "# df = df.loc[df[0] != 0]\n",
    "# f_l = torch.tensor(df[2].values.astype(int))\n",
    "# t_l = torch.tensor(df[3].values.astype(int))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d3de0bc2",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# split_side = int(len(f_l) / batch_edges)\n",
    "# total_edges = np.arange(len(f_l))\n",
    "# np.random.shuffle(total_edges)\n",
    "# new_index = np.array_split(total_edges, split_side)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "8850c1de",
   "metadata": {},
   "outputs": [],
   "source": [
    "changed_list = []\n",
    "changed_list_change = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "d534f1de",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████| 153430/153430 [2:02:42<00:00, 20.84it/s]\n"
     ]
    }
   ],
   "source": [
    "# convert to one-hot labels\n",
    "enc = OneHotEncoder(handle_unknown='ignore')\n",
    "enc.fit(train_y.reshape(-1, 1))\n",
    "one_hot_labels_train_orig = enc.transform(train_y.reshape(-1, 1)).toarray()\n",
    "\n",
    "one_hot_labels_val = enc.transform(val_y.reshape(-1, 1)).toarray()\n",
    "\n",
    "# train the original data\n",
    "# calculate the hessian matrix\n",
    "lr_origin = SimplifiedGraphNeuralNetwork(l2_reg=l2_term, fit_intercept=True)\n",
    "\n",
    "lr_origin.fit(train_x, train_y, sample_weight=None, verbose=False)\n",
    "\n",
    "logits_val_y_origin = val_x @ lr_origin.model.coef_.T + lr_origin.model.intercept_\n",
    "\n",
    "logits_train_y_origin = train_x @ lr_origin.model.coef_.T + lr_origin.model.intercept_\n",
    "\n",
    "ori_val_loss, ave_ori_val_loss = lr_origin.log_loss(logits_val_y_origin, one_hot_labels_val, l2_reg=True)\n",
    "\n",
    "# numpy_theoritic_loss = log_loss(val_y, softmax(logits_val_y_origin, axis=1))\n",
    "\n",
    "# assert np.allclose(numpy_theoritic_loss, ave_ori_val_loss)\n",
    "\n",
    "val_loss_total_grad_orig, val_loss_indiv_grad_orig = lr_origin.grad(val_x, \n",
    "                                                                    logits_val_y_origin,\n",
    "                                                                    one_hot_labels_val, l2_reg = True)\n",
    "\n",
    "hess = lr_origin.hess_cuda(train_x, logits_train_y_origin, l2_reg = True)\n",
    "\n",
    "\n",
    "###### write out hessian matrix\n",
    "\n",
    "hess = cp.asnumpy(hess)\n",
    "\n",
    "pd.DataFrame(hess).to_csv(\"reddit/hess.csv\", index = False)\n",
    "\n",
    "######\n",
    "\n",
    "\n",
    "# loss_grad_hvp = fast_get_inv_hvp_cuda(hess, val_loss_total_grad_orig.T, cholskey=True)\n",
    "\n",
    "# loss_grad_hvp = cp.asnumpy(loss_grad_hvp)\n",
    "del hess"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "2bcd22d8",
   "metadata": {},
   "outputs": [],
   "source": [
    "x_gpu = cp.array([[1, 2, 3], [4, 5, 6]])\n",
    "x_cpu = cp.asnumpy(x_gpu)\n",
    "x_cpu\n",
    "pd.DataFrame(x_cpu).to_csv(\"reddit/file.csv\", index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "cc229beb",
   "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>0</th>\n",
       "      <th>1</th>\n",
       "      <th>2</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>3</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>4</td>\n",
       "      <td>5</td>\n",
       "      <td>6</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   0  1  2\n",
       "0  1  2  3\n",
       "1  4  5  6"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pd.read_csv('reddit/file.csv')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "c8faef04",
   "metadata": {},
   "outputs": [],
   "source": [
    "# # convert to one-hot labels\n",
    "# enc = OneHotEncoder(handle_unknown='ignore')\n",
    "# enc.fit(train_y.reshape(-1, 1))\n",
    "# one_hot_labels_train_orig = enc.transform(train_y.reshape(-1, 1)).toarray()\n",
    "\n",
    "# one_hot_labels_test = enc.transform(test_y.reshape(-1, 1)).toarray()\n",
    "\n",
    "# # train the original data\n",
    "# # calculate the hessian matrix\n",
    "# lr_origin = SimplifiedGraphNeuralNetwork(l2_reg=l2_term, fit_intercept=True)\n",
    "\n",
    "# lr_origin.fit(train_x, train_y, sample_weight=None, verbose=False)\n",
    "\n",
    "# logits_test_y_origin = test_x @ lr_origin.model.coef_.T + lr_origin.model.intercept_\n",
    "\n",
    "# logits_train_y_origin = train_x @ lr_origin.model.coef_.T + lr_origin.model.intercept_\n",
    "\n",
    "# ori_val_loss, ave_ori_val_loss = lr_origin.log_loss(logits_test_y_origin, one_hot_labels_test, l2_reg=True)\n",
    "\n",
    "# # numpy_theoritic_loss = log_loss(test_y, softmax(logits_test_y_origin, axis=1))\n",
    "\n",
    "# # assert np.allclose(numpy_theoritic_loss, ave_ori_val_loss)\n",
    "\n",
    "# val_loss_total_grad_orig, val_loss_indiv_grad_orig = lr_origin.grad(test_x, \n",
    "#                                                                     logits_test_y_origin,\n",
    "#                                                                     one_hot_labels_test, l2_reg = True)\n",
    "\n",
    "# hess = lr_origin.hess_cuda(train_x, logits_train_y_origin, l2_reg = True)\n",
    "\n",
    "# loss_grad_hvp = fast_get_inv_hvp_cuda(hess, val_loss_total_grad_orig.T, cholskey=True)\n",
    "\n",
    "# loss_grad_hvp = cp.asnumpy(loss_grad_hvp)\n",
    "# del hess"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "90afd2bc",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████| 12431/12431 [48:23<00:00,  4.28it/s]\n"
     ]
    }
   ],
   "source": [
    "acctual_influence_1 = []\n",
    "acctual_influence_2 = []\n",
    "\n",
    "predict_influence_1 = []\n",
    "predict_influence_2 = []\n",
    "\n",
    "# for k in tqdm(range(len(train_node_idx))):\n",
    "for k in tqdm(range(len(f_l))):\n",
    "# for k in tqdm(range(len(new_index))):\n",
    "#     eis = EdgeInfluenceSGC(graph=graph, feature=feat, from_index=f_l[new_index[k]], to_index=t_l[new_index[k]])\n",
    "#     eis.remove_edges_sgc_from_influence()\n",
    "    eis = EdgeInfluenceSGC(graph=graph, feature=feat, from_index=f_l[k], to_index=t_l[k])\n",
    "    eis.remove_edges_sgc_from_influence()\n",
    "    feat_removed1 = eis.calculate_modified_features()\n",
    "    \n",
    "#     node_id = train_node_idx.numpy()[k]\n",
    "#     nis = NodeInfluenceSGC(graph = graph, feature=feat, node_index=node_id)\n",
    "#     nis.remove_edges_sgc()\n",
    "#     feat_removed1 = nis.calculate_modified_features()\n",
    "\n",
    "    extra_index = torch.unique(torch.where(feat0 != feat_removed1)[0])\n",
    "    \n",
    "    \n",
    "    extra_index_train = torch.tensor(\n",
    "        [extra_index[i] for i in range(len(extra_index)) if extra_index[i] in train_node_idx]).numpy()\n",
    "    \n",
    "    extra_index_train_in_train = [\n",
    "        np.where(train_node_idx.numpy() == extra_index_train[j])[0][0] for j in range(len(extra_index_train))]\n",
    "    \n",
    "    if extra_index_train == []:\n",
    "        predict_influence_1.append(0.0)\n",
    "        acctual_influence_1.append(0.0)\n",
    "        continue\n",
    "    \n",
    "    feat_to_be_added = feat_removed1[extra_index_train].numpy()\n",
    "    perturb_index = extra_index_train_in_train\n",
    "    \n",
    "    \n",
    "    train_x_new = feat_to_be_added\n",
    "    train_y_new = train_y[perturb_index]\n",
    "    \n",
    "    train_x_orig = np.concatenate([train_x, train_x_new])\n",
    "    train_y_orig = np.concatenate([train_y, train_y_new])\n",
    "    \n",
    "    one_hot_labels_train_0 = enc.transform(train_y_orig.reshape(-1, 1)).toarray()\n",
    "    logits_train_y_origin_0 = train_x_orig @ lr_origin.model.coef_.T + lr_origin.model.intercept_\n",
    "\n",
    "    train_total_grad_orig, train_indiv_grad_orig = lr_origin.grad(train_x_orig, \n",
    "                                            logits_train_y_origin_0, \n",
    "                                            one_hot_labels_train_0, l2_reg = True)\n",
    "    \n",
    "    \n",
    "    pred_infl = train_indiv_grad_orig.dot(loss_grad_hvp)\n",
    "    \n",
    "    \n",
    "    \n",
    "    \n",
    "#     weight_orig = np.ones(len(train_x_orig)) # 1...1...11\n",
    "    \n",
    "#     weight_1 = np.ones(len(train_x_orig))\n",
    "#     weight_1[len(train_x_orig) - len(perturb_index):] = 0 # 1...1...10\n",
    "    \n",
    "#     weight_2 = np.ones(len(train_x_orig))\n",
    "#     weight_2[len(train_x_orig) - len(perturb_index):] = 0 \n",
    "#     weight_2[perturb_index] = 0 # 1...0...10\n",
    "    \n",
    "    weight_3 = np.ones(len(train_x_orig))\n",
    "    weight_3[perturb_index] = 0 # 1...0...11\n",
    "    \n",
    "#     lr_new_1 = SimplifiedGraphNeuralNetwork(l2_reg=1.0, fit_intercept=True)\n",
    "#     train_x_delete_1 = train_x_orig[weight_1 == 1]\n",
    "#     train_y_delete_1 = train_y_orig[weight_1 == 1]\n",
    "    \n",
    "#     assert(np.allclose(train_x_delete_1, train_x))\n",
    "#     assert(np.allclose(train_y_delete_1, train_y))\n",
    "    \n",
    "#     lr_new_1.fit(train_x_delete_1, train_y_delete_1)\n",
    "#     logits_val_y_new_1 = val_x @ lr_new_1.model.coef_.T + lr_new_1.model.intercept_\n",
    "#     new_ori_val_loss_1, _ = lr_new_1.log_loss(logits_val_y_new_1, one_hot_labels_val)\n",
    "    \n",
    "    \n",
    "    \n",
    "    lr_new_2 = SimplifiedGraphNeuralNetwork(l2_reg=l2_term, fit_intercept=True)\n",
    "    train_x_delete_2 = train_x_orig[weight_3 == 1]\n",
    "    train_y_delete_2 = train_y_orig[weight_3 == 1]\n",
    "    \n",
    "    lr_new_2.fit(train_x_delete_2, train_y_delete_2)\n",
    "    logits_val_y_new_2 = val_x @ lr_new_2.model.coef_.T + lr_new_2.model.intercept_\n",
    "    new_ori_val_loss_2, _ = lr_new_2.log_loss(logits_val_y_new_2, one_hot_labels_val, l2_reg = True)\n",
    "    \n",
    "    predict_influence_1.append(np.sum(pred_infl[perturb_index]) - np.sum(pred_infl[len(train_x):]))\n",
    "    acctual_influence_1.append(new_ori_val_loss_2 - ori_val_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5e9d062c",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "7a12d6e1",
   "metadata": {},
   "outputs": [],
   "source": [
    "# acctual_influence_1 = []\n",
    "# acctual_influence_2 = []\n",
    "\n",
    "# predict_influence_1 = []\n",
    "# predict_influence_2 = []\n",
    "\n",
    "# # for k in tqdm(range(len(train_node_idx))):\n",
    "# for k in tqdm(range(len(f_l))):\n",
    "\n",
    "#     eis = EdgeInfluenceSGC(graph=graph, feature=feat, from_index=f_l[k], to_index=t_l[k])\n",
    "#     eis.remove_edges_sgc()\n",
    "#     feat_removed1 = eis.calculate_modified_features()\n",
    "    \n",
    "# #     node_id = train_node_idx.numpy()[k]\n",
    "# #     nis = NodeInfluenceSGC(graph = graph, feature=feat, node_index=node_id)\n",
    "# #     nis.remove_edges_sgc()\n",
    "# #     feat_removed1 = nis.calculate_modified_features()\n",
    "\n",
    "#     extra_index = torch.unique(torch.where(feat0 != feat_removed1)[0])\n",
    "    \n",
    "    \n",
    "#     extra_index_train = torch.tensor(\n",
    "#         [extra_index[i] for i in range(len(extra_index)) if extra_index[i] in train_node_idx]).numpy()\n",
    "    \n",
    "#     extra_index_train_in_train = [\n",
    "#         np.where(train_node_idx.numpy() == extra_index_train[j])[0][0] for j in range(len(extra_index_train))]\n",
    "    \n",
    "#     if extra_index_train != []:\n",
    "#         predict_influence_1.append(0.0)\n",
    "#         acctual_influence_1.append(0.0)\n",
    "#         continue\n",
    "    \n",
    "#     feat_to_be_added = feat_removed1[extra_index_train].numpy()\n",
    "#     perturb_index = extra_index_train_in_train\n",
    "    \n",
    "    \n",
    "#     train_x_new = feat_to_be_added\n",
    "#     train_y_new = train_y[perturb_index]\n",
    "    \n",
    "#     train_x_orig = np.concatenate([train_x, train_x_new])\n",
    "#     train_y_orig = np.concatenate([train_y, train_y_new])\n",
    "    \n",
    "#     one_hot_labels_train_0 = enc.transform(train_y_orig.reshape(-1, 1)).toarray()\n",
    "#     logits_train_y_origin_0 = train_x_orig @ lr_origin.model.coef_.T + lr_origin.model.intercept_\n",
    "\n",
    "#     train_total_grad_orig, train_indiv_grad_orig = lr_origin.grad(train_x_orig, \n",
    "#                                             logits_train_y_origin_0, \n",
    "#                                             one_hot_labels_train_0, l2_reg = True)\n",
    "    \n",
    "    \n",
    "#     pred_infl = train_indiv_grad_orig.dot(loss_grad_hvp)\n",
    "    \n",
    "    \n",
    "    \n",
    "    \n",
    "# #     weight_orig = np.ones(len(train_x_orig)) # 1...1...11\n",
    "    \n",
    "# #     weight_1 = np.ones(len(train_x_orig))\n",
    "# #     weight_1[len(train_x_orig) - len(perturb_index):] = 0 # 1...1...10\n",
    "    \n",
    "# #     weight_2 = np.ones(len(train_x_orig))\n",
    "# #     weight_2[len(train_x_orig) - len(perturb_index):] = 0 \n",
    "# #     weight_2[perturb_index] = 0 # 1...0...10\n",
    "    \n",
    "#     weight_3 = np.ones(len(train_x_orig))\n",
    "#     weight_3[perturb_index] = 0 # 1...0...11\n",
    "    \n",
    "# #     lr_new_1 = SimplifiedGraphNeuralNetwork(l2_reg=1.0, fit_intercept=True)\n",
    "# #     train_x_delete_1 = train_x_orig[weight_1 == 1]\n",
    "# #     train_y_delete_1 = train_y_orig[weight_1 == 1]\n",
    "    \n",
    "# #     assert(np.allclose(train_x_delete_1, train_x))\n",
    "# #     assert(np.allclose(train_y_delete_1, train_y))\n",
    "    \n",
    "# #     lr_new_1.fit(train_x_delete_1, train_y_delete_1)\n",
    "# #     logits_test_y_new_1 = test_x @ lr_new_1.model.coef_.T + lr_new_1.model.intercept_\n",
    "# #     new_ori_val_loss_1, _ = lr_new_1.log_loss(logits_test_y_new_1, one_hot_labels_test)\n",
    "    \n",
    "    \n",
    "    \n",
    "#     lr_new_2 = SimplifiedGraphNeuralNetwork(l2_reg=l2_term, fit_intercept=True)\n",
    "#     train_x_delete_2 = train_x_orig[weight_3 == 1]\n",
    "#     train_y_delete_2 = train_y_orig[weight_3 == 1]\n",
    "    \n",
    "#     lr_new_2.fit(train_x_delete_2, train_y_delete_2)\n",
    "#     logits_test_y_new_2 = test_x @ lr_new_2.model.coef_.T + lr_new_2.model.intercept_\n",
    "#     new_ori_val_loss_2, _ = lr_new_2.log_loss(logits_test_y_new_2, one_hot_labels_test, l2_reg = True)\n",
    "    \n",
    "#     predict_influence_1.append(np.sum(pred_infl[perturb_index]) - np.sum(pred_infl[len(train_x):]))\n",
    "#     acctual_influence_1.append(new_ori_val_loss_2 - ori_val_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "83390a35",
   "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"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of edges in consider 12431\n"
     ]
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "low_limit = -10\n",
    "up_limit = 10\n",
    "x = np.linspace(low_limit, up_limit)\n",
    "# x = np.linspace(-0.2, 0.15)\n",
    "plt.plot(x, x)\n",
    "plt.scatter(acctual_influence_1, predict_influence_1, s = 10, color='blue')\n",
    "plt.xlabel('actual change in loss')\n",
    "plt.ylabel('predicted change in loss')\n",
    "# plt.title('Influence function on edges of Cora dataset perturb edges:' + str(batch_edges))\n",
    "# plt.title('Influence function on Complete Node of Citeseer dataset')\n",
    "plt.ylim(low_limit, up_limit)\n",
    "plt.xlim(low_limit, up_limit)\n",
    "# plt.title('Influence function on Iris dataset')\n",
    "plt.show()\n",
    "print('Number of edges in consider %d'% len(f_l))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "cfcf7118",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([33])"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "extra_index_train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "6f305a71",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([ 0.82303029,  0.01698285,  0.03169363, ..., -0.19991923,\n",
       "       -0.10448183,  0.28174796])"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.array(acctual_influence_1)[np.array(acctual_influence_1) !=0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "8617e601",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "SpearmanrResult(correlation=0.6023873823666681, pvalue=0.0)"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "scipy.stats.spearmanr(np.array(acctual_influence_1)[np.array(acctual_influence_1) !=0], np.array(predict_influence_1)[np.array(predict_influence_1) !=0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "13c3758a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "SpearmanrResult(correlation=0.4882905335821803, pvalue=0.0)"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "scipy.stats.spearmanr(acctual_influence_1, predict_influence_1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "2dd7e551",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.DataFrame([acctual_influence_1, predict_influence_1, f_l.numpy().astype(int), t_l.numpy().astype(int)]).T\n",
    "df.columns = ['acctual_influence', 'predict_influence', 'from_edges', 'to_edges']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "5c999518",
   "metadata": {},
   "outputs": [],
   "source": [
    "df.to_csv('result_data/' + data_set + '_edge_influence_001.csv', header = None, index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "1a75b1b5",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "df = pd.read_csv('result_data/pubmed_edge_influence.csv', header = None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "984f20d5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x7fddb56c9df0>"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "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",
    "import numpy as np\n",
    "low_limit = -10\n",
    "up_limit = 10\n",
    "x = np.linspace(low_limit, up_limit)\n",
    "# x = np.linspace(-0.2, 0.15)\n",
    "plt.plot(x, x)\n",
    "plt.scatter(df[0], df[1], s = 10, color='blue')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "401141ed",
   "metadata": {},
   "outputs": [],
   "source": [
    "g = dgl.graph(([1, 1, 2, 3, 4], [0, 2, 0, 4, 2]))\n",
    "g.edata['w'] = torch.arange(10).view(5, 2)\n",
    "sg, inverse_indices = dgl.khop_in_subgraph(g, 0, k=2)"
   ]
  }
 ],
 "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
}
