{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "23ab6f4b",
   "metadata": {},
   "outputs": [],
   "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 torch.nn.functional as F\n",
    "import scipy\n",
    "import deeprobust\n",
    "from dgl import function as fn\n",
    "from dgl.base import DGLError\n",
    "from dataset import load_graph_dataset\n",
    "from matplotlib import pyplot as plt\n",
    "from tqdm import tqdm\n",
    "from scipy import sparse\n",
    "from deeprobust.graph.data import Dataset\n",
    "from deeprobust.graph.defense import GCN\n",
    "from deeprobust.graph.global_attack import Metattack, NodeEmbeddingAttack"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "cbd877de",
   "metadata": {},
   "outputs": [],
   "source": [
    "# data = Dataset(root='/tmp/', name='cora', setting='nettack')\n",
    "# adj, features, labels = data.adj, data.features, data.labels\n",
    "# idx_train, idx_val, idx_test = data.idx_train, data.idx_val, data.idx_test\n",
    "# idx_unlabeled = np.union1d(idx_val, idx_test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "22e1f467",
   "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"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zizhang/anaconda3/lib/python3.8/site-packages/dgl/heterograph.py:3719: DGLWarning: DGLGraph.adjacency_matrix_scipy is deprecated. Please replace it with:\n",
      "\n",
      "\tDGLGraph.adjacency_matrix(transpose, scipy_fmt=\"csr\").\n",
      "\n",
      "  dgl_warning('DGLGraph.adjacency_matrix_scipy is deprecated. '\n"
     ]
    }
   ],
   "source": [
    "data_set = 'cora'\n",
    "graph, feat, labels, train_mask, val_mask, test_mask, number_classes = load_graph_dataset(data_set)\n",
    "\n",
    "adj = graph.adjacency_matrix_scipy()\n",
    "features = sparse.csr_matrix(feat.numpy())\n",
    "labels = labels.numpy().astype(int)\n",
    "idx_train = np.where(train_mask == 1)[0]\n",
    "idx_val = np.where(val_mask == 1)[0]\n",
    "idx_test = np.where(test_mask == 1)[0]\n",
    "idx_unlabeled = np.union1d(idx_val, idx_test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "df7dba74",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "13264"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "graph.number_of_edges()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "70dc4a29",
   "metadata": {},
   "outputs": [],
   "source": [
    "# device = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")\n",
    "# surrogate = GCN(nfeat=features.shape[1], nclass=labels.max().item()+1, nhid=16, dropout=0.0,\n",
    "#                 with_relu=False, device=device)\n",
    "# surrogate = surrogate.to(device)\n",
    "# surrogate.fit(features, adj, labels, idx_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "379e2023",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model = Metattack(model=surrogate, nnodes=adj.shape[0], feature_shape=features.shape, device=device)\n",
    "# model = model.to(device)\n",
    "# perturbations = int(0.1 * (adj.sum() // 2))\n",
    "# model.attack(features, adj, labels, idx_train, idx_test, perturbations, ll_constraint=False)\n",
    "# modified_adj = model.modified_adj"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b44aa02f",
   "metadata": {},
   "source": [
    "##### 1, DICE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "eea682c9",
   "metadata": {},
   "outputs": [],
   "source": [
    "import random\n",
    "import numpy as np\n",
    "import scipy.sparse as sp\n",
    "from deeprobust.graph.global_attack import BaseAttack\n",
    "\n",
    "class DICE(BaseAttack):\n",
    "    \"\"\"As is described in ADVERSARIAL ATTACKS ON GRAPH NEURAL NETWORKS VIA META LEARNING (ICLR'19),\n",
    "    'DICE (delete internally, connect externally) is a baseline where, for each perturbation,\n",
    "    we randomly choose whether to insert or remove an edge. Edges are only removed between\n",
    "    nodes from the same classes, and only inserted between nodes from different classes.\n",
    "    Parameters\n",
    "    ----------\n",
    "    model :\n",
    "        model to attack. Default `None`.\n",
    "    nnodes : int\n",
    "        number of nodes in the input graph\n",
    "    attack_structure : bool\n",
    "        whether to attack graph structure\n",
    "    attack_features : bool\n",
    "        whether to attack node features\n",
    "    device: str\n",
    "        'cpu' or 'cuda'\n",
    "    Examples\n",
    "    --------\n",
    "    >>> from deeprobust.graph.data import Dataset\n",
    "    >>> from deeprobust.graph.global_attack import DICE\n",
    "    >>> data = Dataset(root='/tmp/', name='cora')\n",
    "    >>> adj, features, labels = data.adj, data.features, data.labels\n",
    "    >>> model = DICE()\n",
    "    >>> model.attack(adj, labels, n_perturbations=10)\n",
    "    >>> modified_adj = model.modified_adj\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self, model=None, nnodes=None, attack_structure=True, attack_features=False, device='cpu'):\n",
    "        super(DICE, self).__init__(model, nnodes, attack_structure=attack_structure, attack_features=attack_features, device=device)\n",
    "\n",
    "        assert not self.attack_features, 'DICE does NOT support attacking features'\n",
    "\n",
    "    def attack(self, ori_adj, labels, n_perturbations, **kwargs):\n",
    "        \"\"\"Delete internally, connect externally. This baseline has all true class labels\n",
    "        (train and test) available.\n",
    "        Parameters\n",
    "        ----------\n",
    "        ori_adj : scipy.sparse.csr_matrix\n",
    "            Original (unperturbed) adjacency matrix.\n",
    "        labels:\n",
    "            node labels\n",
    "        n_perturbations : int\n",
    "            Number of edge removals/additions.\n",
    "        Returns\n",
    "        -------\n",
    "        None.\n",
    "        \"\"\"\n",
    "\n",
    "        # ori_adj: sp.csr_matrix\n",
    "\n",
    "        print('number of pertubations: %s' % n_perturbations)\n",
    "        modified_adj = ori_adj.tolil()\n",
    "\n",
    "#         remove_or_insert = np.random.choice(2, n_perturbations)\n",
    "        n_remove = n_perturbations\n",
    "\n",
    "        nonzero = set(zip(*ori_adj.nonzero()))\n",
    "        indices = sp.triu(modified_adj).nonzero()\n",
    "        possible_indices = [x for x in zip(indices[0], indices[1])\n",
    "                            if labels[x[0]] == labels[x[1]]]\n",
    "\n",
    "        remove_indices = np.random.permutation(possible_indices)[: n_remove]\n",
    "        modified_adj[remove_indices[:, 0], remove_indices[:, 1]] = 0\n",
    "        modified_adj[remove_indices[:, 1], remove_indices[:, 0]] = 0\n",
    "\n",
    "#         n_insert = n_perturbations - n_remove\n",
    "        n_insert = 0\n",
    "        # sample edges to add\n",
    "        added_edges = 0\n",
    "        while added_edges < n_insert:\n",
    "            n_remaining = n_insert - added_edges\n",
    "\n",
    "            # sample random pairs\n",
    "            candidate_edges = np.array([np.random.choice(ori_adj.shape[0], n_remaining),\n",
    "                                        np.random.choice(ori_adj.shape[0], n_remaining)]).T\n",
    "\n",
    "            # filter out existing edges, and pairs with the different labels\n",
    "            candidate_edges = set([(u, v) for u, v in candidate_edges if labels[u] != labels[v]\n",
    "                                        and modified_adj[u, v] == 0 and modified_adj[v, u] == 0])\n",
    "            candidate_edges = np.array(list(candidate_edges))\n",
    "\n",
    "            # if none is found, try again\n",
    "            if len(candidate_edges) == 0:\n",
    "                continue\n",
    "\n",
    "            # add all found edges to your modified adjacency matrix\n",
    "            modified_adj[candidate_edges[:, 0], candidate_edges[:, 1]] = 1\n",
    "            modified_adj[candidate_edges[:, 1], candidate_edges[:, 0]] = 1\n",
    "            added_edges += candidate_edges.shape[0]\n",
    "\n",
    "        self.check_adj(modified_adj)\n",
    "        self.modified_adj = modified_adj\n",
    "\n",
    "\n",
    "    def sample_forever(self, adj, exclude):\n",
    "        \"\"\"Randomly random sample edges from adjacency matrix, `exclude` is a set\n",
    "        which contains the edges we do not want to sample and the ones already sampled\n",
    "        \"\"\"\n",
    "        while True:\n",
    "            # t = tuple(np.random.randint(0, adj.shape[0], 2))\n",
    "            t = tuple(random.sample(range(0, adj.shape[0]), 2))\n",
    "            if t not in exclude:\n",
    "                yield t\n",
    "                exclude.add(t)\n",
    "                exclude.add((t[1], t[0]))\n",
    "\n",
    "    def random_sample_edges(self, adj, n, exclude):\n",
    "        itr = self.sample_forever(adj, exclude=exclude)\n",
    "        return [next(itr) for _ in range(n)]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "59b1ca36",
   "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_set = 'citeseer'\n",
    "data_set = 'cora'\n",
    "graph, feat, labels, train_mask, val_mask, test_mask, number_classes = load_graph_dataset(data_set)\n",
    "\n",
    "adj = graph.adjacency_matrix_scipy()\n",
    "features = sparse.csr_matrix(feat.numpy())\n",
    "labels = labels.numpy().astype(int)\n",
    "idx_train = np.where(train_mask == 1)[0]\n",
    "idx_val = np.where(val_mask == 1)[0]\n",
    "idx_test = np.where(test_mask == 1)[0]\n",
    "idx_unlabeled = np.union1d(idx_val, idx_test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "40948104",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of pertubations: 52\n"
     ]
    }
   ],
   "source": [
    "rate = 0.01\n",
    "model = DICE()\n",
    "adj_0 = adj - np.eye(adj.shape[0])\n",
    "adj_0 = sparse.csr_matrix(adj_0)\n",
    "\n",
    "total_edges = int(np.sum(adj_0) / 2)\n",
    "n_perturbation = int(rate * total_edges)\n",
    "\n",
    "model.attack(adj_0, labels, n_perturbations = n_perturbation)\n",
    "modified_adj = model.modified_adj\n",
    "\n",
    "f_l = np.where(adj_0.toarray() != modified_adj.toarray())[0]\n",
    "t_l = np.where(adj_0.toarray() != modified_adj.toarray())[1]\n",
    "df =  pd.DataFrame([f_l.astype(int), t_l.astype(int)]).T\n",
    "df.to_csv('DICE/new_ver' + data_set + '_remove_rate_' +str(rate) + '.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "17a5ba8d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(104,)"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.where(modified_adj.toarray() - adj_0.toarray() < 0)[0].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "70a11e0e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of pertubations: 52\n",
      "number of pertubations: 158\n",
      "number of pertubations: 263\n"
     ]
    }
   ],
   "source": [
    "rate = [0.01, 0.03, 0.05]\n",
    "\n",
    "for i in range(len(rate)):\n",
    "\n",
    "    model = DICE()\n",
    "    adj_0 = adj - np.eye(adj.shape[0])\n",
    "    adj_0 = sparse.csr_matrix(adj_0)\n",
    "\n",
    "    total_edges = int(np.sum(adj_0) / 2)\n",
    "    n_perturbation = int(rate[i] * total_edges)\n",
    "\n",
    "    model.attack(adj_0, labels, n_perturbations = n_perturbation)\n",
    "    modified_adj = model.modified_adj\n",
    "\n",
    "    f_l = np.where(adj_0.toarray() != modified_adj.toarray())[0]\n",
    "    t_l = np.where(adj_0.toarray() != modified_adj.toarray())[1]\n",
    "    df =  pd.DataFrame([f_l.astype(int), t_l.astype(int)]).T\n",
    "    df.to_csv('DICE/new_' + data_set + '_remove_rate_' +str(rate[i]) + '.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "ffc9837c",
   "metadata": {},
   "outputs": [],
   "source": [
    "f_l = np.where(adj_0.toarray() != modified_adj.toarray())[0]\n",
    "t_l = np.where(adj_0.toarray() != modified_adj.toarray())[1]\n",
    "df =  pd.DataFrame([f_l.astype(int), t_l.astype(int)]).T\n",
    "df.to_csv('DICE/' + data_set + '_remove_rate_' +str(rate[0]) + '.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "ee24c919",
   "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",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1</td>\n",
       "      <td>5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>5</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>5</td>\n",
       "      <td>6</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>5</td>\n",
       "      <td>7</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>6</td>\n",
       "      <td>5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>521</th>\n",
       "      <td>2697</td>\n",
       "      <td>2702</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>522</th>\n",
       "      <td>2700</td>\n",
       "      <td>2693</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>523</th>\n",
       "      <td>2701</td>\n",
       "      <td>2703</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>524</th>\n",
       "      <td>2702</td>\n",
       "      <td>2697</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>525</th>\n",
       "      <td>2703</td>\n",
       "      <td>2701</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>526 rows × 2 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "        0     1\n",
       "0       1     5\n",
       "1       5     1\n",
       "2       5     6\n",
       "3       5     7\n",
       "4       6     5\n",
       "..    ...   ...\n",
       "521  2697  2702\n",
       "522  2700  2693\n",
       "523  2701  2703\n",
       "524  2702  2697\n",
       "525  2703  2701\n",
       "\n",
       "[526 rows x 2 columns]"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pd.read_csv('DICE/cora_remove_rate_0.01.csv')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f049a43a",
   "metadata": {},
   "source": [
    "##### Node embedding attack"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "68f78b7b",
   "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",
      "52\n"
     ]
    }
   ],
   "source": [
    "data_set = 'cora'\n",
    "rate = 0.01\n",
    "\n",
    "graph, feat, labels, train_mask, val_mask, test_mask, number_classes = load_graph_dataset(data_set)\n",
    "\n",
    "adj = graph.adjacency_matrix_scipy()\n",
    "features = sparse.csr_matrix(feat.numpy())\n",
    "labels = labels.numpy().astype(int)\n",
    "idx_train = np.where(train_mask == 1)[0]\n",
    "idx_val = np.where(val_mask == 1)[0]\n",
    "idx_test = np.where(test_mask == 1)[0]\n",
    "idx_unlabeled = np.union1d(idx_val, idx_test)\n",
    "\n",
    "adj_0 = adj - np.eye(adj.shape[0])\n",
    "\n",
    "adj_0 = sparse.csr_matrix(adj_0)\n",
    "\n",
    "total_edges = int(np.sum(adj_0) / 2)\n",
    "\n",
    "n_perturbation = int(rate * total_edges)\n",
    "model = NodeEmbeddingAttack()\n",
    "\n",
    "model.attack(adj_0, attack_type=\"remove\", n_perturbations = n_perturbation)\n",
    "modified_adj = model.modified_adj\n",
    "\n",
    "print(n_perturbation)\n",
    "model.attack(adj_0, attack_type=\"remove\", min_span_tree=True, n_perturbations = n_perturbation)\n",
    "modified_adj = model.modified_adj\n",
    "\n",
    "# model.attack(adj, attack_type=\"add\", n_candidates=10000)\n",
    "# modified_adj = model.modified_adj\n",
    "# model.attack(adj, attack_type=\"add_by_remove\", n_candidates=10000)\n",
    "# modified_adj = model.modified_adj\n",
    "\n",
    "f_l = np.where(adj_0.toarray() != modified_adj.toarray())[0]\n",
    "t_l = np.where(adj_0.toarray() != modified_adj.toarray())[1]\n",
    "df =  pd.DataFrame([f_l.astype(int), t_l.astype(int)]).T\n",
    "df.to_csv('NodeEmbeddingAttack/new' + data_set + '_remove_rate_' +str(rate) + '.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "a28b7da4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[1, 1, 1, ..., 0, 0, 0],\n",
       "       [1, 1, 0, ..., 0, 0, 0],\n",
       "       [1, 0, 1, ..., 0, 0, 0],\n",
       "       ...,\n",
       "       [0, 0, 0, ..., 1, 0, 0],\n",
       "       [0, 0, 0, ..., 0, 1, 1],\n",
       "       [0, 0, 0, ..., 0, 1, 1]])"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "adj_0.toarray()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "534b9b4d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((104, 2), 52)"
      ]
     },
     "execution_count": 39,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df.shape, n_perturbation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "718ce09d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(104,)"
      ]
     },
     "execution_count": 40,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ids = np.where(adj_0.toarray() - modified_adj.toarray() > 0)\n",
    "f_id = ids[0]\n",
    "t_id = ids[1]\n",
    "f_id.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7fd65e8e",
   "metadata": {},
   "outputs": [],
   "source": [
    "f_l = np.where(adj_0.toarray() != modified_adj.toarray())[0]\n",
    "t_l = np.where(adj_0.toarray() != modified_adj.toarray())[1]\n",
    "df =  pd.DataFrame([f_l.astype(int), t_l.astype(int)]).T\n",
    "df.to_csv('NodeEmbeddingAttack/' + data_set + '_remove_rate_' +str(rate[0]) + '.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3a109f3b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "07cae2f9",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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
}
