{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "89c78cc9",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Using backend: pytorch\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import copy\n",
    "import torch\n",
    "import dgl\n",
    "import cupy as cp\n",
    "import collections\n",
    "import random\n",
    "from dgl import function as fn\n",
    "from sklearn.preprocessing import OneHotEncoder\n",
    "from dataset import load_graph_dataset\n",
    "from tqdm import tqdm\n",
    "from model_softmax import SimplifiedGraphNeuralNetwork\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "\n",
    "from torch_geometric.datasets import Planetoid \n",
    "from pygod.utils import gen_attribute_outliers\n",
    "from pygod.models import CoLA, DOMINANT\n",
    "from dgl.data import FraudDataset\n",
    "from pygod.utils.metric import eval_roc_auc, eval_recall_at_k, eval_precision_at_k\n",
    "from sklearn.metrics import precision_score, accuracy_score\n",
    "from model_softmax import SimplifiedGraphNeuralNetwork, fast_hess, fast_hess_cuda, fast_get_inv_hvp_cuda\n",
    "from tqdm import tqdm\n",
    "from sklearn.metrics.pairwise import euclidean_distances\n",
    "from sklearn.neighbors import NearestCentroid\n",
    "\n",
    "from outlier_generator import inject_edge_outlier, generate_node_feature_outliers\n",
    "from model_edge_influence import EdgeInfluenceSGC\n",
    "\n",
    "import itertools"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "ee6b7da4",
   "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": "903ab1ca",
   "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": [
    "data_dgl = 'cora'\n",
    "graph, feat, labels, train_mask, val_mask, test_mask, number_classes = load_graph_dataset(data_dgl)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "2e0907e1",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_graph, edge_oulier_labels = inject_edge_outlier(graph, labels,num_edges=126)\n",
    "graph = new_graph"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "453af0c5",
   "metadata": {},
   "outputs": [],
   "source": [
    "# l2_term = 0.01\n",
    "l2_term = 0.01\n",
    "# batch_edges = 1\n",
    "num_layer = 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "007453df",
   "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": 10,
   "id": "250fe1c0",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_x = feat0.numpy().astype(np.float32)\n",
    "train_y = labels.numpy().astype(np.float32)\n",
    "\n",
    "val_x = feat0.numpy().astype(np.float32)\n",
    "val_y = labels.numpy().astype(np.float32)\n",
    "\n",
    "train_node_idx = graph.nodes()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "20be62fe",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "' Train Logistic Regression '"
      ]
     },
     "execution_count": 13,
     "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": 14,
   "id": "98e077af",
   "metadata": {},
   "outputs": [],
   "source": [
    "from_indexes, to_indexes = graph.edges()\n",
    "\n",
    "f_l, t_l = from_indexes, to_indexes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "296057c3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "13516"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(from_indexes)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "4d20a470",
   "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_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",
    "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": 12,
   "id": "34d4dc8e",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████| 13516/13516 [30:15<00:00,  7.44it/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": 40,
   "id": "7ad9222b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "27e24b15",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([13516])"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "new_graph.edges()[0].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "aa1a2f64",
   "metadata": {},
   "outputs": [
    {
     "ename": "ValueError",
     "evalue": "x and y must be the same size",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mValueError\u001b[0m                                Traceback (most recent call last)",
      "\u001b[0;32m/tmp/ipykernel_5059/2629886871.py\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      5\u001b[0m \u001b[0;31m# x = np.linspace(-0.2, 0.15)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      6\u001b[0m \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mplot\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mx\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 7\u001b[0;31m \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mscatter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0medge_oulier_labels\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mpredict_influence_1\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0ms\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;36m10\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcolor\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m'blue'\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      8\u001b[0m \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mxlabel\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'actual change in loss'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      9\u001b[0m \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mylabel\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'predicted change in loss'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/anaconda3/lib/python3.8/site-packages/matplotlib/pyplot.py\u001b[0m in \u001b[0;36mscatter\u001b[0;34m(x, y, s, c, marker, cmap, norm, vmin, vmax, alpha, linewidths, edgecolors, plotnonfinite, data, **kwargs)\u001b[0m\n\u001b[1;32m   3066\u001b[0m         \u001b[0mvmin\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mNone\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mvmax\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mNone\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0malpha\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mNone\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mlinewidths\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mNone\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[0m\n\u001b[1;32m   3067\u001b[0m         edgecolors=None, plotnonfinite=False, data=None, **kwargs):\n\u001b[0;32m-> 3068\u001b[0;31m     __ret = gca().scatter(\n\u001b[0m\u001b[1;32m   3069\u001b[0m         \u001b[0mx\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0ms\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0ms\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mc\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mc\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmarker\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mmarker\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcmap\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mcmap\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mnorm\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mnorm\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   3070\u001b[0m         \u001b[0mvmin\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mvmin\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mvmax\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mvmax\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0malpha\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0malpha\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mlinewidths\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mlinewidths\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/anaconda3/lib/python3.8/site-packages/matplotlib/__init__.py\u001b[0m in \u001b[0;36minner\u001b[0;34m(ax, data, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1359\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0minner\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0max\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdata\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mNone\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\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   1360\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mdata\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1361\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mfunc\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0max\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0mmap\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0msanitize_sequence\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0margs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\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   1362\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1363\u001b[0m         \u001b[0mbound\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnew_sig\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbind\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0max\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/anaconda3/lib/python3.8/site-packages/matplotlib/axes/_axes.py\u001b[0m in \u001b[0;36mscatter\u001b[0;34m(self, x, y, s, c, marker, cmap, norm, vmin, vmax, alpha, linewidths, edgecolors, plotnonfinite, **kwargs)\u001b[0m\n\u001b[1;32m   4496\u001b[0m         \u001b[0my\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mma\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mravel\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0my\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   4497\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mx\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msize\u001b[0m \u001b[0;34m!=\u001b[0m \u001b[0my\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msize\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 4498\u001b[0;31m             \u001b[0;32mraise\u001b[0m \u001b[0mValueError\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m\"x and y must be the same size\"\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   4499\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   4500\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0ms\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mValueError\u001b[0m: x and y must be the same size"
     ]
    },
    {
     "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",
    "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(edge_oulier_labels, 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": 46,
   "id": "6887239f",
   "metadata": {},
   "outputs": [],
   "source": [
    "old_graph = graph.clone()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "05a0424c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "13264"
      ]
     },
     "execution_count": 51,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(graph.edges()[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "id": "8408ca3b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([0, 0, 0, ..., 0, 0, 0])"
      ]
     },
     "execution_count": 54,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.concatenate([np.repeat(0, 13264), np.repeat(1, 252)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "id": "1753a72c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,\n",
       "       1, 1, 1, 1, 1, 1, 1, 1, 1, 1])"
      ]
     },
     "execution_count": 56,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "56e1a01f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "13516"
      ]
     },
     "execution_count": 42,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(predict_influence_1)"
   ]
  }
 ],
 "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
}
