{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 55,
   "id": "459b02ef",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import itertools\n",
    "import random\n",
    "\n",
    "import graph_tool.all as gt\n",
    "from graph_tool.spectral import adjacency\n",
    "from src.gwi.geometric_weights_inference import geometric_weights_inference"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "id": "6e0d9b72",
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy.sparse import csr_matrix\n",
    "\n",
    "\n",
    "def _sparse_estimator_wrapper(x: csr_matrix, estimator: str, **kwargs):\n",
    "    return getattr(x, estimator)(**kwargs).A1\n",
    "\n",
    "\n",
    "def _sparse_sum(x: csr_matrix, **kwargs):\n",
    "    return _sparse_estimator_wrapper(x, \"sum\", **kwargs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 57,
   "id": "dfd04003",
   "metadata": {},
   "outputs": [],
   "source": [
    "base_path = '../dataset/ATLAS'\n",
    "network = 'covert_FIFA2015'\n",
    "network_path = network + '.gt'\n",
    "method_name = 'CND'\n",
    "top_node = 0.01\n",
    "add_link = 0.01"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1305a4c8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Graph loaded successfully!\n",
      "File: ../dataset/ATLAS/covert_FIFA2015.gt\n",
      "Number of vertices (nodes): 414\n",
      "Number of edges (links): 4805\n",
      "Computing weighting for current_weighting='CND' took 0.00 s\n",
      "Identified the top 1.0% of nodes.\n",
      "Total nodes in graph: 414\n",
      "Number of top nodes selected: 4\n",
      "Indices of the top nodes: \n",
      "[ 53 157 127  92]\n",
      "Created a copy of the graph. Initial edges: 4805\n",
      "Node 53 has 3475 missing links to add with 96 neighbors.\n",
      "Node 157 has 3490 missing links to add with 96 neighbors.\n",
      "Node 127 has 4282 missing links to add with 105 neighbors.\n",
      "Node 92 has 5155 missing links to add with 115 neighbors.\n",
      "\n",
      "Finished processing all top nodes.\n",
      "Total new links added: 161\n",
      "Nodes in original graph 'g':      414\n",
      "Edges in original graph 'g':      4805\n",
      "Nodes in new robust graph 'g_robust': 414\n",
      "Edges in new robust graph 'g_robust': 4966\n",
      "\n",
      "Successfully saved the robust graph to: ../dataset/ATLAS/covert_FIFA2015_robust_CND_nodes1p_links1p.gt\n"
     ]
    }
   ],
   "source": [
    "filename = os.path.join(base_path, network_path) \n",
    "\n",
    "g = gt.Graph()\n",
    "g.load(filename)\n",
    "\n",
    "print(\"Graph loaded successfully!\")\n",
    "print(f\"File: {filename}\")\n",
    "print(f\"Number of vertices (nodes): {g.num_vertices()}\")\n",
    "print(f\"Number of edges (links): {g.num_edges()}\")\n",
    "\n",
    "\n",
    "network_result = geometric_weights_inference(\n",
    "    graph=g,\n",
    "    weightings=[method_name],\n",
    ")\n",
    "\n",
    "node_values = _sparse_sum(network_result.weighting_results[method_name], axis=1)\n",
    "\n",
    "num_nodes = g.num_vertices()\n",
    "\n",
    "top_k_count = max(1, int(num_nodes * top_node))\n",
    "\n",
    "sorted_indices = np.argsort(node_values)\n",
    "\n",
    "top_nodes_indices = sorted_indices[-top_k_count:]\n",
    "\n",
    "print(f\"Identified the top {top_node*100}% of nodes.\")\n",
    "print(f\"Total nodes in graph: {num_nodes}\")\n",
    "print(f\"Number of top nodes selected: {len(top_nodes_indices)}\")\n",
    "print(f\"Indices of the top nodes: \\n{top_nodes_indices}\")\n",
    "\n",
    "\n",
    "g_robust = g.copy()\n",
    "print(f\"Created a copy of the graph. Initial edges: {g_robust.num_edges()}\")\n",
    "\n",
    "total_links_added = 0\n",
    "\n",
    "for node_index in top_nodes_indices:\n",
    "    target_node = g.vertex(node_index)\n",
    "\n",
    "    neighbors = list(target_node.out_neighbors())\n",
    "\n",
    "    if len(neighbors) < 2:\n",
    "        continue\n",
    "\n",
    "    missing_links = []\n",
    "    for u, v in itertools.combinations(neighbors, 2):\n",
    "\n",
    "        if g_robust.edge(u, v) is None:\n",
    "            missing_links.append((u, v))\n",
    "    \n",
    "    print(f\"Node {node_index} has {len(missing_links)} missing links to add with {len(neighbors)} neighbors.\")\n",
    "    \n",
    "    if not missing_links:\n",
    "        continue\n",
    "\n",
    "    num_to_add = max(1, int(len(missing_links) * add_link))\n",
    "    \n",
    "    num_to_add = min(num_to_add, len(missing_links))\n",
    "\n",
    "    links_to_add = random.sample(missing_links, num_to_add)\n",
    "\n",
    "    for u, v in links_to_add:\n",
    "        new_edge = g_robust.add_edge(u, v)\n",
    "        \n",
    "        if 'weight' in g_robust.ep:\n",
    "            g_robust.ep.weight[new_edge] = 1.0\n",
    "    \n",
    "    total_links_added += len(links_to_add)\n",
    "\n",
    "print(f\"\\nFinished processing all top nodes.\")\n",
    "print(f\"Total new links added: {total_links_added}\")\n",
    "print(f\"Nodes in original graph 'g':      {g.num_vertices()}\")\n",
    "print(f\"Edges in original graph 'g':      {g.num_edges()}\")\n",
    "print(f\"Nodes in new robust graph 'g_robust': {g_robust.num_vertices()}\")\n",
    "print(f\"Edges in new robust graph 'g_robust': {g_robust.num_edges()}\")\n",
    "\n",
    "top_nodes_p_str = int(top_node * 100)\n",
    "links_added_p_str = int(add_link * 100)\n",
    "\n",
    "adjacency_matrix = adjacency(g_robust, csr=True).astype(\n",
    "copy=False, dtype=np.float64\n",
    ")\n",
    "if adjacency_matrix.diagonal().sum() != 0:\n",
    "    raise ValueError(\"No self-loops are allowed\")\n",
    "\n",
    "if adjacency_matrix.min() < 0 or adjacency_matrix.max() > 1:\n",
    "    print(adjacency_matrix.min(), adjacency_matrix.max())\n",
    "    raise ValueError(\"Network must be unweighted\")\n",
    "\n",
    "\n",
    "new_filename = f\"{base_path}/{network}_robust_{method_name}_nodes{str(top_nodes_p_str)}p_links{links_added_p_str}p.gt\"\n",
    "\n",
    "g_robust.save(new_filename)\n",
    "print(f\"\\nSuccessfully saved the robust graph to: {new_filename}\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "ra-dismantling-full",
   "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.12.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
