

<!DOCTYPE html>
<html class="writer-html5" lang="en" >
<head>
  <meta charset="utf-8">
  
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  
  <title>deeprobust.graph.targeted_attack.nettack &mdash; DeepRobust 0.1.1 documentation</title>
  

  
  <link rel="stylesheet" href="../../../../_static/css/theme.css" type="text/css" />
  <link rel="stylesheet" href="../../../../_static/pygments.css" type="text/css" />

  
  
  
  

  
  <!--[if lt IE 9]>
    <script src="../../../../_static/js/html5shiv.min.js"></script>
  <![endif]-->
  
    
      <script type="text/javascript" id="documentation_options" data-url_root="../../../../" src="../../../../_static/documentation_options.js"></script>
        <script type="text/javascript" src="../../../../_static/jquery.js"></script>
        <script type="text/javascript" src="../../../../_static/underscore.js"></script>
        <script type="text/javascript" src="../../../../_static/doctools.js"></script>
        <script type="text/javascript" src="../../../../_static/language_data.js"></script>
        <script async="async" type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.5/latest.js?config=TeX-AMS-MML_HTMLorMML"></script>
    
    <script type="text/javascript" src="../../../../_static/js/theme.js"></script>

    
    <link rel="index" title="Index" href="../../../../genindex.html" />
    <link rel="search" title="Search" href="../../../../search.html" /> 
</head>

<body class="wy-body-for-nav">

   
  <div class="wy-grid-for-nav">
    
    <nav data-toggle="wy-nav-shift" class="wy-nav-side">
      <div class="wy-side-scroll">
        <div class="wy-side-nav-search" >
          

          
            <a href="../../../../index.html" class="icon icon-home" alt="Documentation Home"> DeepRobust
          

          
          </a>

          
            
            
          

          
<div role="search">
  <form id="rtd-search-form" class="wy-form" action="../../../../search.html" method="get">
    <input type="text" name="q" placeholder="Search docs" />
    <input type="hidden" name="check_keywords" value="yes" />
    <input type="hidden" name="area" value="default" />
  </form>
</div>

          
        </div>

        
        <div class="wy-menu wy-menu-vertical" data-spy="affix" role="navigation" aria-label="main navigation">
          
            
            
              
            
            
              <p class="caption"><span class="caption-text">Installation</span></p>
<ul>
<li class="toctree-l1"><a class="reference internal" href="../../../../notes/installation.html">Installation</a></li>
</ul>
<p class="caption"><span class="caption-text">Graph Package</span></p>
<ul>
<li class="toctree-l1"><a class="reference internal" href="../../../../graph/data.html">Graph Dataset</a></li>
<li class="toctree-l1"><a class="reference internal" href="../../../../graph/attack.html">Introduction to Graph Attack with Examples</a></li>
<li class="toctree-l1"><a class="reference internal" href="../../../../graph/defense.html">Introduction to Graph Defense with Examples</a></li>
<li class="toctree-l1"><a class="reference internal" href="../../../../graph/pyg.html">Using PyTorch Geometric in DeepRobust</a></li>
<li class="toctree-l1"><a class="reference internal" href="../../../../graph/node_embedding.html">Node Embedding Attack and Defense</a></li>
</ul>
<p class="caption"><span class="caption-text">Image Package</span></p>
<ul>
<li class="toctree-l1"><a class="reference internal" href="../../../../image/example.html">Image Attack and Defense</a></li>
</ul>
<p class="caption"><span class="caption-text">Image Package</span></p>
<ul>
<li class="toctree-l1"><a class="reference internal" href="../../../../source/deeprobust.image.attack.html">deeprobust.image.attack package</a></li>
<li class="toctree-l1"><a class="reference internal" href="../../../../source/deeprobust.image.defense.html">deeprobust.image.defense package</a></li>
<li class="toctree-l1"><a class="reference internal" href="../../../../source/deeprobust.image.netmodels.html">deeprobust.image.netmodels package</a></li>
</ul>
<p class="caption"><span class="caption-text">Graph Package</span></p>
<ul>
<li class="toctree-l1"><a class="reference internal" href="../../../../source/deeprobust.graph.global_attack.html">deeprobust.graph.global_attack package</a></li>
<li class="toctree-l1"><a class="reference internal" href="../../../../source/deeprobust.graph.targeted_attack.html">deeprobust.graph.targeted_attack package</a></li>
<li class="toctree-l1"><a class="reference internal" href="../../../../source/deeprobust.graph.defense.html">deeprobust.graph.defense package</a></li>
<li class="toctree-l1"><a class="reference internal" href="../../../../source/deeprobust.graph.data.html">deeprobust.graph.data package</a></li>
</ul>

            
          
        </div>
        
      </div>
    </nav>

    <section data-toggle="wy-nav-shift" class="wy-nav-content-wrap">

      
      <nav class="wy-nav-top" aria-label="top navigation">
        
          <i data-toggle="wy-nav-top" class="fa fa-bars"></i>
          <a href="../../../../index.html">DeepRobust</a>
        
      </nav>


      <div class="wy-nav-content">
        
        <div class="rst-content">
        
          















<div role="navigation" aria-label="breadcrumbs navigation">

  <ul class="wy-breadcrumbs">
    
      <li><a href="../../../../index.html" class="icon icon-home"></a> &raquo;</li>
        
          <li><a href="../../../index.html">Module code</a> &raquo;</li>
        
      <li>deeprobust.graph.targeted_attack.nettack</li>
    
    
      <li class="wy-breadcrumbs-aside">
        
      </li>
    
  </ul>

  
  <hr/>
</div>
          <div role="main" class="document" itemscope="itemscope" itemtype="http://schema.org/Article">
           <div itemprop="articleBody">
            
  <h1>Source code for deeprobust.graph.targeted_attack.nettack</h1><div class="highlight"><pre>
<span></span><span class="sd">&quot;&quot;&quot;</span>
<span class="sd">    Adversarial Attacks on Neural Networks for Graph Data. KDD 2018.</span>
<span class="sd">        https://arxiv.org/pdf/1805.07984.pdf</span>
<span class="sd">    Author&#39;s Implementation</span>
<span class="sd">        https://github.com/danielzuegner/nettack</span>

<span class="sd">    Since pytorch does not have good enough support to the operations</span>
<span class="sd">    on sparse tensor, this part of code is heavily based on the author&#39;s implementation.</span>
<span class="sd">&quot;&quot;&quot;</span>
<span class="sd">&quot;&quot;&quot;</span>
<span class="sd">Implementation of the method proposed in the paper:</span>
<span class="sd">&#39;Adversarial Attacks on Neural Networks for Graph Data&#39;</span>
<span class="sd">by Daniel Zügner, Amir Akbarnejad and Stephan Günnemann,</span>
<span class="sd">published at SIGKDD&#39;18, August 2018, London, UK</span>
<span class="sd">Copyright (C) 2018</span>
<span class="sd">Daniel Zügner</span>
<span class="sd">Technical University of Munich</span>
<span class="sd">&quot;&quot;&quot;</span>

<span class="kn">import</span> <span class="nn">torch</span>
<span class="kn">from</span> <span class="nn">deeprobust.graph.targeted_attack</span> <span class="kn">import</span> <span class="n">BaseAttack</span>
<span class="kn">from</span> <span class="nn">torch.nn.parameter</span> <span class="kn">import</span> <span class="n">Parameter</span>
<span class="kn">from</span> <span class="nn">deeprobust.graph</span> <span class="kn">import</span> <span class="n">utils</span>
<span class="kn">import</span> <span class="nn">torch.nn.functional</span> <span class="k">as</span> <span class="nn">F</span>
<span class="kn">from</span> <span class="nn">torch</span> <span class="kn">import</span> <span class="n">optim</span>
<span class="kn">from</span> <span class="nn">torch.nn</span> <span class="kn">import</span> <span class="n">functional</span> <span class="k">as</span> <span class="n">F</span>
<span class="kn">from</span> <span class="nn">torch.nn.modules.module</span> <span class="kn">import</span> <span class="n">Module</span>
<span class="kn">from</span> <span class="nn">torch.nn.parameter</span> <span class="kn">import</span> <span class="n">Parameter</span>
<span class="kn">import</span> <span class="nn">numpy</span> <span class="k">as</span> <span class="nn">np</span>
<span class="kn">import</span> <span class="nn">scipy.sparse</span> <span class="k">as</span> <span class="nn">sp</span>
<span class="kn">from</span> <span class="nn">copy</span> <span class="kn">import</span> <span class="n">deepcopy</span>
<span class="kn">from</span> <span class="nn">numba</span> <span class="kn">import</span> <span class="n">jit</span>
<span class="kn">from</span> <span class="nn">torch</span> <span class="kn">import</span> <span class="n">spmm</span>

<div class="viewcode-block" id="Nettack"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.Nettack">[docs]</a><span class="k">class</span> <span class="nc">Nettack</span><span class="p">(</span><span class="n">BaseAttack</span><span class="p">):</span>
    <span class="sd">&quot;&quot;&quot;Nettack.</span>

<span class="sd">    Parameters</span>
<span class="sd">    ----------</span>
<span class="sd">    model :</span>
<span class="sd">        model to attack</span>
<span class="sd">    nnodes : int</span>
<span class="sd">        number of nodes in the input graph</span>
<span class="sd">    attack_structure : bool</span>
<span class="sd">        whether to attack graph structure</span>
<span class="sd">    attack_features : bool</span>
<span class="sd">        whether to attack node features</span>
<span class="sd">    device: str</span>
<span class="sd">        &#39;cpu&#39; or &#39;cuda&#39;</span>

<span class="sd">    Examples</span>
<span class="sd">    --------</span>

<span class="sd">    &gt;&gt;&gt; from deeprobust.graph.data import Dataset</span>
<span class="sd">    &gt;&gt;&gt; from deeprobust.graph.defense import GCN</span>
<span class="sd">    &gt;&gt;&gt; from deeprobust.graph.targeted_attack import Nettack</span>
<span class="sd">    &gt;&gt;&gt; data = Dataset(root=&#39;/tmp/&#39;, name=&#39;cora&#39;)</span>
<span class="sd">    &gt;&gt;&gt; adj, features, labels = data.adj, data.features, data.labels</span>
<span class="sd">    &gt;&gt;&gt; idx_train, idx_val, idx_test = data.idx_train, data.idx_val, data.idx_test</span>
<span class="sd">    &gt;&gt;&gt; # Setup Surrogate model</span>
<span class="sd">    &gt;&gt;&gt; surrogate = GCN(nfeat=features.shape[1], nclass=labels.max().item()+1,</span>
<span class="sd">                    nhid=16, dropout=0, with_relu=False, with_bias=False, device=&#39;cpu&#39;).to(&#39;cpu&#39;)</span>
<span class="sd">    &gt;&gt;&gt; surrogate.fit(features, adj, labels, idx_train, idx_val, patience=30)</span>
<span class="sd">    &gt;&gt;&gt; # Setup Attack Model</span>
<span class="sd">    &gt;&gt;&gt; target_node = 0</span>
<span class="sd">    &gt;&gt;&gt; model = Nettack(surrogate, nnodes=adj.shape[0], attack_structure=True, attack_features=True, device=&#39;cpu&#39;).to(&#39;cpu&#39;)</span>
<span class="sd">    &gt;&gt;&gt; # Attack</span>
<span class="sd">    &gt;&gt;&gt; model.attack(features, adj, labels, target_node, n_perturbations=5)</span>
<span class="sd">    &gt;&gt;&gt; modified_adj = model.modified_adj</span>
<span class="sd">    &gt;&gt;&gt; modified_features = model.modified_features</span>

<span class="sd">    &quot;&quot;&quot;</span>

    <span class="k">def</span> <span class="fm">__init__</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">model</span><span class="p">,</span> <span class="n">nnodes</span><span class="o">=</span><span class="kc">None</span><span class="p">,</span> <span class="n">attack_structure</span><span class="o">=</span><span class="kc">True</span><span class="p">,</span> <span class="n">attack_features</span><span class="o">=</span><span class="kc">False</span><span class="p">,</span> <span class="n">device</span><span class="o">=</span><span class="s1">&#39;cpu&#39;</span><span class="p">):</span>

        <span class="nb">super</span><span class="p">(</span><span class="n">Nettack</span><span class="p">,</span> <span class="bp">self</span><span class="p">)</span><span class="o">.</span><span class="fm">__init__</span><span class="p">(</span><span class="n">model</span><span class="p">,</span> <span class="n">nnodes</span><span class="p">,</span> <span class="n">attack_structure</span><span class="o">=</span><span class="n">attack_structure</span><span class="p">,</span> <span class="n">attack_features</span><span class="o">=</span><span class="n">attack_features</span><span class="p">,</span> <span class="n">device</span><span class="o">=</span><span class="n">device</span><span class="p">)</span>

        <span class="bp">self</span><span class="o">.</span><span class="n">structure_perturbations</span> <span class="o">=</span> <span class="p">[]</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">feature_perturbations</span> <span class="o">=</span> <span class="p">[]</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span> <span class="o">=</span> <span class="p">[]</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">potential_edges</span> <span class="o">=</span> <span class="p">[]</span>

        <span class="bp">self</span><span class="o">.</span><span class="n">cooc_constraint</span> <span class="o">=</span> <span class="kc">None</span>

<div class="viewcode-block" id="Nettack.filter_potential_singletons"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.Nettack.filter_potential_singletons">[docs]</a>    <span class="k">def</span> <span class="nf">filter_potential_singletons</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">modified_adj</span><span class="p">):</span>
        <span class="sd">&quot;&quot;&quot;Computes a mask for entries potentially leading to singleton nodes, i.e.</span>
<span class="sd">        one of the two nodes corresponding to the entry have degree 1 and there</span>
<span class="sd">        is an edge between the two nodes.</span>
<span class="sd">        &quot;&quot;&quot;</span>

        <span class="n">degrees</span> <span class="o">=</span> <span class="n">modified_adj</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="mi">0</span><span class="p">)</span>
        <span class="n">degree_one</span> <span class="o">=</span> <span class="p">(</span><span class="n">degrees</span> <span class="o">==</span> <span class="mi">1</span><span class="p">)</span>
        <span class="n">resh</span> <span class="o">=</span> <span class="n">degree_one</span><span class="o">.</span><span class="n">repeat</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="p">,</span> <span class="mi">1</span><span class="p">)</span><span class="o">.</span><span class="n">float</span><span class="p">()</span>
        <span class="n">l_and</span> <span class="o">=</span> <span class="n">resh</span> <span class="o">*</span> <span class="n">modified_adj</span>
        <span class="n">logical_and_symmetric</span> <span class="o">=</span> <span class="n">l_and</span> <span class="o">+</span> <span class="n">l_and</span><span class="o">.</span><span class="n">t</span><span class="p">()</span>
        <span class="n">flat_mask</span> <span class="o">=</span> <span class="mi">1</span> <span class="o">-</span> <span class="n">logical_and_symmetric</span>
        <span class="k">return</span> <span class="n">flat_mask</span></div>

    <span class="k">def</span> <span class="nf">get_linearized_weight</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
        <span class="n">surrogate</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">surrogate</span>
        <span class="n">W</span> <span class="o">=</span> <span class="n">surrogate</span><span class="o">.</span><span class="n">gc1</span><span class="o">.</span><span class="n">weight</span> <span class="o">@</span> <span class="n">surrogate</span><span class="o">.</span><span class="n">gc2</span><span class="o">.</span><span class="n">weight</span>
        <span class="k">return</span> <span class="n">W</span><span class="o">.</span><span class="n">detach</span><span class="p">()</span><span class="o">.</span><span class="n">cpu</span><span class="p">()</span><span class="o">.</span><span class="n">numpy</span><span class="p">()</span>

<div class="viewcode-block" id="Nettack.attack"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.Nettack.attack">[docs]</a>    <span class="k">def</span> <span class="nf">attack</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">features</span><span class="p">,</span> <span class="n">adj</span><span class="p">,</span> <span class="n">labels</span><span class="p">,</span> <span class="n">target_node</span><span class="p">,</span> <span class="n">n_perturbations</span><span class="p">,</span> <span class="n">direct</span><span class="o">=</span><span class="kc">True</span><span class="p">,</span> <span class="n">n_influencers</span><span class="o">=</span> <span class="mi">0</span><span class="p">,</span> <span class="n">ll_cutoff</span><span class="o">=</span><span class="mf">0.004</span><span class="p">,</span> <span class="n">verbose</span><span class="o">=</span><span class="kc">True</span><span class="p">,</span> <span class="o">**</span><span class="n">kwargs</span><span class="p">):</span>
        <span class="sd">&quot;&quot;&quot;Generate perturbations on the input graph.</span>

<span class="sd">        Parameters</span>
<span class="sd">        ----------</span>
<span class="sd">        ori_features : torch.Tensor or scipy.sparse.csr_matrix</span>
<span class="sd">            Origina (unperturbed) node feature matrix. Note that</span>
<span class="sd">            torch.Tensor will be automatically transformed into</span>
<span class="sd">            scipy.sparse.csr_matrix</span>
<span class="sd">        ori_adj : torch.Tensor or scipy.sparse.csr_matrix</span>
<span class="sd">            Original (unperturbed) adjacency matrix. Note that</span>
<span class="sd">            torch.Tensor will be automatically transformed into</span>
<span class="sd">            scipy.sparse.csr_matrix</span>
<span class="sd">        labels :</span>
<span class="sd">            node labels</span>
<span class="sd">        target_node : int</span>
<span class="sd">            target node index to be attacked</span>
<span class="sd">        n_perturbations : int</span>
<span class="sd">            Number of perturbations on the input graph. Perturbations could</span>
<span class="sd">            be edge removals/additions or feature removals/additions.</span>
<span class="sd">        direct: bool</span>
<span class="sd">            whether to conduct direct attack</span>
<span class="sd">        n_influencers:</span>
<span class="sd">            number of influencer nodes when performing indirect attack.</span>
<span class="sd">            (setting `direct` to False). When `direct` is True, it would be ignored.</span>
<span class="sd">        ll_cutoff : float</span>
<span class="sd">            The critical value for the likelihood ratio test of the power law distributions.</span>
<span class="sd">            See the Chi square distribution with one degree of freedom. Default value 0.004</span>
<span class="sd">            corresponds to a p-value of roughly 0.95.</span>
<span class="sd">        verbose : bool</span>
<span class="sd">            whether to show verbose logs</span>
<span class="sd">        &quot;&quot;&quot;</span>

        <span class="k">if</span> <span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span> <span class="ow">is</span> <span class="kc">None</span><span class="p">:</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span> <span class="o">=</span> <span class="n">adj</span><span class="o">.</span><span class="n">shape</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span>

        <span class="bp">self</span><span class="o">.</span><span class="n">target_node</span> <span class="o">=</span> <span class="n">target_node</span>

        <span class="k">if</span> <span class="nb">type</span><span class="p">(</span><span class="n">adj</span><span class="p">)</span> <span class="ow">is</span> <span class="n">torch</span><span class="o">.</span><span class="n">Tensor</span><span class="p">:</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">ori_adj</span> <span class="o">=</span> <span class="n">utils</span><span class="o">.</span><span class="n">to_scipy</span><span class="p">(</span><span class="n">adj</span><span class="p">)</span><span class="o">.</span><span class="n">tolil</span><span class="p">()</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span> <span class="o">=</span> <span class="n">utils</span><span class="o">.</span><span class="n">to_scipy</span><span class="p">(</span><span class="n">adj</span><span class="p">)</span><span class="o">.</span><span class="n">tolil</span><span class="p">()</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">ori_features</span> <span class="o">=</span> <span class="n">utils</span><span class="o">.</span><span class="n">to_scipy</span><span class="p">(</span><span class="n">features</span><span class="p">)</span><span class="o">.</span><span class="n">tolil</span><span class="p">()</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span> <span class="o">=</span> <span class="n">utils</span><span class="o">.</span><span class="n">to_scipy</span><span class="p">(</span><span class="n">features</span><span class="p">)</span><span class="o">.</span><span class="n">tolil</span><span class="p">()</span>
        <span class="k">else</span><span class="p">:</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">ori_adj</span> <span class="o">=</span> <span class="n">adj</span><span class="o">.</span><span class="n">tolil</span><span class="p">()</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span> <span class="o">=</span> <span class="n">adj</span><span class="o">.</span><span class="n">tolil</span><span class="p">()</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">ori_features</span> <span class="o">=</span> <span class="n">features</span><span class="o">.</span><span class="n">tolil</span><span class="p">()</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span> <span class="o">=</span> <span class="n">features</span><span class="o">.</span><span class="n">tolil</span><span class="p">()</span>

        <span class="bp">self</span><span class="o">.</span><span class="n">cooc_matrix</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="o">.</span><span class="n">T</span><span class="o">.</span><span class="n">dot</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="p">)</span><span class="o">.</span><span class="n">tolil</span><span class="p">()</span>

        <span class="n">attack_features</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">attack_features</span>
        <span class="n">attack_structure</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">attack_structure</span>
        <span class="k">assert</span> <span class="ow">not</span> <span class="p">(</span><span class="n">direct</span><span class="o">==</span><span class="kc">False</span> <span class="ow">and</span> <span class="n">n_influencers</span><span class="o">==</span><span class="mi">0</span><span class="p">),</span> <span class="s2">&quot;indirect mode requires at least one influencer node&quot;</span>
        <span class="k">assert</span> <span class="n">n_perturbations</span> <span class="o">&gt;</span> <span class="mi">0</span><span class="p">,</span> <span class="s2">&quot;need at least one perturbation&quot;</span>
        <span class="k">assert</span> <span class="n">attack_features</span> <span class="ow">or</span> <span class="n">attack_structure</span><span class="p">,</span> <span class="s2">&quot;either attack_features or attack_structure must be true&quot;</span>

        <span class="c1"># adj_norm = utils.normalize_adj_tensor(modified_adj, sparse=True)</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span> <span class="o">=</span> <span class="n">utils</span><span class="o">.</span><span class="n">normalize_adj</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="p">)</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">W</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">get_linearized_weight</span><span class="p">()</span>

        <span class="n">logits</span> <span class="o">=</span> <span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span> <span class="o">@</span> <span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span> <span class="o">@</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span> <span class="o">@</span> <span class="bp">self</span><span class="o">.</span><span class="n">W</span> <span class="p">)[</span><span class="n">target_node</span><span class="p">]</span>

        <span class="bp">self</span><span class="o">.</span><span class="n">label_u</span> <span class="o">=</span> <span class="n">labels</span><span class="p">[</span><span class="n">target_node</span><span class="p">]</span>
        <span class="n">label_target_onehot</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">eye</span><span class="p">(</span><span class="nb">int</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">nclass</span><span class="p">))[</span><span class="n">labels</span><span class="p">[</span><span class="n">target_node</span><span class="p">]]</span>
        <span class="n">best_wrong_class</span> <span class="o">=</span> <span class="p">(</span><span class="n">logits</span> <span class="o">-</span> <span class="mi">1000</span><span class="o">*</span><span class="n">label_target_onehot</span><span class="p">)</span><span class="o">.</span><span class="n">argmax</span><span class="p">()</span>
        <span class="n">surrogate_losses</span> <span class="o">=</span> <span class="p">[</span><span class="n">logits</span><span class="p">[</span><span class="n">labels</span><span class="p">[</span><span class="n">target_node</span><span class="p">]]</span> <span class="o">-</span> <span class="n">logits</span><span class="p">[</span><span class="n">best_wrong_class</span><span class="p">]]</span>

        <span class="k">if</span> <span class="n">verbose</span><span class="p">:</span>
            <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;##### Starting attack #####&quot;</span><span class="p">)</span>
            <span class="k">if</span> <span class="n">attack_structure</span> <span class="ow">and</span> <span class="n">attack_features</span><span class="p">:</span>
                <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;##### Attack node with ID </span><span class="si">{}</span><span class="s2"> using structure and feature perturbations #####&quot;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">target_node</span><span class="p">))</span>
            <span class="k">elif</span> <span class="n">attack_features</span><span class="p">:</span>
                <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;##### Attack only using feature perturbations #####&quot;</span><span class="p">)</span>
            <span class="k">elif</span> <span class="n">attack_structure</span><span class="p">:</span>
                <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;##### Attack only using structure perturbations #####&quot;</span><span class="p">)</span>
            <span class="k">if</span> <span class="n">direct</span><span class="p">:</span>
                <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;##### Attacking the node directly #####&quot;</span><span class="p">)</span>
            <span class="k">else</span><span class="p">:</span>
                <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;##### Attacking the node indirectly via </span><span class="si">{}</span><span class="s2"> influencer nodes #####&quot;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">n_influencers</span><span class="p">))</span>
            <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;##### Performing </span><span class="si">{}</span><span class="s2"> perturbations #####&quot;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">n_perturbations</span><span class="p">))</span>

        <span class="k">if</span> <span class="n">attack_structure</span><span class="p">:</span>
            <span class="c1"># Setup starting values of the likelihood ratio test.</span>
            <span class="n">degree_sequence_start</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">ori_adj</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="mi">0</span><span class="p">)</span><span class="o">.</span><span class="n">A1</span>
            <span class="n">current_degree_sequence</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="mi">0</span><span class="p">)</span><span class="o">.</span><span class="n">A1</span>
            <span class="n">d_min</span> <span class="o">=</span> <span class="mi">2</span>

            <span class="n">S_d_start</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">log</span><span class="p">(</span><span class="n">degree_sequence_start</span><span class="p">[</span><span class="n">degree_sequence_start</span> <span class="o">&gt;=</span> <span class="n">d_min</span><span class="p">]))</span>
            <span class="n">current_S_d</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">log</span><span class="p">(</span><span class="n">current_degree_sequence</span><span class="p">[</span><span class="n">current_degree_sequence</span> <span class="o">&gt;=</span> <span class="n">d_min</span><span class="p">]))</span>
            <span class="n">n_start</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="n">degree_sequence_start</span> <span class="o">&gt;=</span> <span class="n">d_min</span><span class="p">)</span>
            <span class="n">current_n</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="n">current_degree_sequence</span> <span class="o">&gt;=</span> <span class="n">d_min</span><span class="p">)</span>
            <span class="n">alpha_start</span> <span class="o">=</span> <span class="n">compute_alpha</span><span class="p">(</span><span class="n">n_start</span><span class="p">,</span> <span class="n">S_d_start</span><span class="p">,</span> <span class="n">d_min</span><span class="p">)</span>

            <span class="n">log_likelihood_orig</span> <span class="o">=</span> <span class="n">compute_log_likelihood</span><span class="p">(</span><span class="n">n_start</span><span class="p">,</span> <span class="n">alpha_start</span><span class="p">,</span> <span class="n">S_d_start</span><span class="p">,</span> <span class="n">d_min</span><span class="p">)</span>

        <span class="k">if</span> <span class="nb">len</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span><span class="p">)</span> <span class="o">==</span> <span class="mi">0</span><span class="p">:</span>
            <span class="k">if</span> <span class="ow">not</span> <span class="n">direct</span><span class="p">:</span>
                <span class="c1"># Choose influencer nodes</span>
                <span class="n">infls</span><span class="p">,</span> <span class="n">add_infls</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">get_attacker_nodes</span><span class="p">(</span><span class="n">n_influencers</span><span class="p">,</span> <span class="n">add_additional_nodes</span><span class="o">=</span><span class="kc">True</span><span class="p">)</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">concatenate</span><span class="p">((</span><span class="n">infls</span><span class="p">,</span> <span class="n">add_infls</span><span class="p">))</span><span class="o">.</span><span class="n">astype</span><span class="p">(</span><span class="s2">&quot;int&quot;</span><span class="p">)</span>
                <span class="c1"># Potential edges are all edges from any attacker to any other node, except the respective</span>
                <span class="c1"># attacker itself or the node being attacked.</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">potential_edges</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">row_stack</span><span class="p">([</span><span class="n">np</span><span class="o">.</span><span class="n">column_stack</span><span class="p">((</span><span class="n">np</span><span class="o">.</span><span class="n">tile</span><span class="p">(</span><span class="n">infl</span><span class="p">,</span> <span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span> <span class="o">-</span> <span class="mi">2</span><span class="p">),</span>
                                                            <span class="n">np</span><span class="o">.</span><span class="n">setdiff1d</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">arange</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="p">),</span>
                                                            <span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">([</span><span class="n">target_node</span><span class="p">,</span><span class="n">infl</span><span class="p">]))))</span> <span class="k">for</span> <span class="n">infl</span> <span class="ow">in</span>
                                                            <span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span><span class="p">])</span>
                <span class="k">if</span> <span class="n">verbose</span><span class="p">:</span>
                    <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;Influencer nodes: </span><span class="si">{}</span><span class="s2">&quot;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span><span class="p">))</span>
            <span class="k">else</span><span class="p">:</span>
                <span class="c1"># direct attack</span>
                <span class="n">influencers</span> <span class="o">=</span> <span class="p">[</span><span class="n">target_node</span><span class="p">]</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">potential_edges</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">column_stack</span><span class="p">((</span><span class="n">np</span><span class="o">.</span><span class="n">tile</span><span class="p">(</span><span class="n">target_node</span><span class="p">,</span> <span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="o">-</span><span class="mi">1</span><span class="p">),</span> <span class="n">np</span><span class="o">.</span><span class="n">setdiff1d</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">arange</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="p">),</span> <span class="n">target_node</span><span class="p">)))</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">(</span><span class="n">influencers</span><span class="p">)</span>

        <span class="bp">self</span><span class="o">.</span><span class="n">potential_edges</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">potential_edges</span><span class="o">.</span><span class="n">astype</span><span class="p">(</span><span class="s2">&quot;int32&quot;</span><span class="p">)</span>

        <span class="k">for</span> <span class="n">_</span> <span class="ow">in</span> <span class="nb">range</span><span class="p">(</span><span class="n">n_perturbations</span><span class="p">):</span>
            <span class="k">if</span> <span class="n">verbose</span><span class="p">:</span>
                <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;##### ...</span><span class="si">{}</span><span class="s2">/</span><span class="si">{}</span><span class="s2"> perturbations ... #####&quot;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">_</span><span class="o">+</span><span class="mi">1</span><span class="p">,</span> <span class="n">n_perturbations</span><span class="p">))</span>
            <span class="k">if</span> <span class="n">attack_structure</span><span class="p">:</span>

                <span class="c1"># Do not consider edges that, if removed, result in singleton edges in the graph.</span>
                <span class="n">singleton_filter</span> <span class="o">=</span> <span class="n">filter_singletons</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">potential_edges</span><span class="p">,</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="p">)</span>
                <span class="n">filtered_edges</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">potential_edges</span><span class="p">[</span><span class="n">singleton_filter</span><span class="p">]</span>

                <span class="c1"># Update the values for the power law likelihood ratio test.</span>

                <span class="n">deltas</span> <span class="o">=</span> <span class="mi">2</span> <span class="o">*</span> <span class="p">(</span><span class="mi">1</span> <span class="o">-</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="p">[</span><span class="nb">tuple</span><span class="p">(</span><span class="n">filtered_edges</span><span class="o">.</span><span class="n">T</span><span class="p">)]</span><span class="o">.</span><span class="n">toarray</span><span class="p">()[</span><span class="mi">0</span><span class="p">]</span> <span class="p">)</span><span class="o">-</span> <span class="mi">1</span>
                <span class="n">d_edges_old</span> <span class="o">=</span> <span class="n">current_degree_sequence</span><span class="p">[</span><span class="n">filtered_edges</span><span class="p">]</span>
                <span class="n">d_edges_new</span> <span class="o">=</span> <span class="n">current_degree_sequence</span><span class="p">[</span><span class="n">filtered_edges</span><span class="p">]</span> <span class="o">+</span> <span class="n">deltas</span><span class="p">[:,</span> <span class="kc">None</span><span class="p">]</span>
                <span class="n">new_S_d</span><span class="p">,</span> <span class="n">new_n</span> <span class="o">=</span> <span class="n">update_Sx</span><span class="p">(</span><span class="n">current_S_d</span><span class="p">,</span> <span class="n">current_n</span><span class="p">,</span> <span class="n">d_edges_old</span><span class="p">,</span> <span class="n">d_edges_new</span><span class="p">,</span> <span class="n">d_min</span><span class="p">)</span>
                <span class="n">new_alphas</span> <span class="o">=</span> <span class="n">compute_alpha</span><span class="p">(</span><span class="n">new_n</span><span class="p">,</span> <span class="n">new_S_d</span><span class="p">,</span> <span class="n">d_min</span><span class="p">)</span>
                <span class="n">new_ll</span> <span class="o">=</span> <span class="n">compute_log_likelihood</span><span class="p">(</span><span class="n">new_n</span><span class="p">,</span> <span class="n">new_alphas</span><span class="p">,</span> <span class="n">new_S_d</span><span class="p">,</span> <span class="n">d_min</span><span class="p">)</span>
                <span class="n">alphas_combined</span> <span class="o">=</span> <span class="n">compute_alpha</span><span class="p">(</span><span class="n">new_n</span> <span class="o">+</span> <span class="n">n_start</span><span class="p">,</span> <span class="n">new_S_d</span> <span class="o">+</span> <span class="n">S_d_start</span><span class="p">,</span> <span class="n">d_min</span><span class="p">)</span>
                <span class="n">new_ll_combined</span> <span class="o">=</span> <span class="n">compute_log_likelihood</span><span class="p">(</span><span class="n">new_n</span> <span class="o">+</span> <span class="n">n_start</span><span class="p">,</span> <span class="n">alphas_combined</span><span class="p">,</span> <span class="n">new_S_d</span> <span class="o">+</span> <span class="n">S_d_start</span><span class="p">,</span> <span class="n">d_min</span><span class="p">)</span>
                <span class="n">new_ratios</span> <span class="o">=</span> <span class="o">-</span><span class="mi">2</span> <span class="o">*</span> <span class="n">new_ll_combined</span> <span class="o">+</span> <span class="mi">2</span> <span class="o">*</span> <span class="p">(</span><span class="n">new_ll</span> <span class="o">+</span> <span class="n">log_likelihood_orig</span><span class="p">)</span>

                <span class="c1"># Do not consider edges that, if added/removed, would lead to a violation of the</span>
                <span class="c1"># likelihood ration Chi_square cutoff value.</span>
                <span class="n">powerlaw_filter</span> <span class="o">=</span> <span class="n">filter_chisquare</span><span class="p">(</span><span class="n">new_ratios</span><span class="p">,</span> <span class="n">ll_cutoff</span><span class="p">)</span>
                <span class="n">filtered_edges_final</span> <span class="o">=</span> <span class="n">filtered_edges</span><span class="p">[</span><span class="n">powerlaw_filter</span><span class="p">]</span>

                <span class="c1"># Compute new entries in A_hat_square_uv</span>
                <span class="n">a_hat_uv_new</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">compute_new_a_hat_uv</span><span class="p">(</span><span class="n">filtered_edges_final</span><span class="p">,</span> <span class="n">target_node</span><span class="p">)</span>
                <span class="c1"># Compute the struct scores for each potential edge</span>
                <span class="n">struct_scores</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">struct_score</span><span class="p">(</span><span class="n">a_hat_uv_new</span><span class="p">,</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span> <span class="o">@</span> <span class="bp">self</span><span class="o">.</span><span class="n">W</span><span class="p">)</span>
                <span class="n">best_edge_ix</span> <span class="o">=</span> <span class="n">struct_scores</span><span class="o">.</span><span class="n">argmin</span><span class="p">()</span>
                <span class="n">best_edge_score</span> <span class="o">=</span> <span class="n">struct_scores</span><span class="o">.</span><span class="n">min</span><span class="p">()</span>
                <span class="n">best_edge</span> <span class="o">=</span> <span class="n">filtered_edges_final</span><span class="p">[</span><span class="n">best_edge_ix</span><span class="p">]</span>

            <span class="k">if</span> <span class="n">attack_features</span><span class="p">:</span>
                <span class="c1"># Compute the feature scores for each potential feature perturbation</span>
                <span class="n">feature_ixs</span><span class="p">,</span> <span class="n">feature_scores</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">feature_scores</span><span class="p">()</span>
                <span class="n">best_feature_ix</span> <span class="o">=</span> <span class="n">feature_ixs</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span>
                <span class="n">best_feature_score</span> <span class="o">=</span> <span class="n">feature_scores</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span>

            <span class="k">if</span> <span class="n">attack_structure</span> <span class="ow">and</span> <span class="n">attack_features</span><span class="p">:</span>
                <span class="c1"># decide whether to choose an edge or feature to change</span>
                <span class="k">if</span> <span class="n">best_edge_score</span> <span class="o">&lt;</span> <span class="n">best_feature_score</span><span class="p">:</span>
                    <span class="k">if</span> <span class="n">verbose</span><span class="p">:</span>
                        <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;Edge perturbation: </span><span class="si">{}</span><span class="s2">&quot;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">best_edge</span><span class="p">))</span>
                    <span class="n">change_structure</span> <span class="o">=</span> <span class="kc">True</span>
                <span class="k">else</span><span class="p">:</span>
                    <span class="k">if</span> <span class="n">verbose</span><span class="p">:</span>
                        <span class="nb">print</span><span class="p">(</span><span class="s2">&quot;Feature perturbation: </span><span class="si">{}</span><span class="s2">&quot;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">best_feature_ix</span><span class="p">))</span>
                    <span class="n">change_structure</span><span class="o">=</span><span class="kc">False</span>

            <span class="k">elif</span> <span class="n">attack_structure</span><span class="p">:</span>
                <span class="n">change_structure</span> <span class="o">=</span> <span class="kc">True</span>
            <span class="k">elif</span> <span class="n">attack_features</span><span class="p">:</span>
                <span class="n">change_structure</span> <span class="o">=</span> <span class="kc">False</span>

            <span class="k">if</span> <span class="n">change_structure</span><span class="p">:</span>
                <span class="c1"># perform edge perturbation</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="p">[</span><span class="nb">tuple</span><span class="p">(</span><span class="n">best_edge</span><span class="p">)]</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="p">[</span><span class="nb">tuple</span><span class="p">(</span><span class="n">best_edge</span><span class="p">[::</span><span class="o">-</span><span class="mi">1</span><span class="p">])]</span> <span class="o">=</span> <span class="mi">1</span> <span class="o">-</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="p">[</span><span class="nb">tuple</span><span class="p">(</span><span class="n">best_edge</span><span class="p">)]</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span> <span class="o">=</span> <span class="n">utils</span><span class="o">.</span><span class="n">normalize_adj</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="p">)</span>

                <span class="bp">self</span><span class="o">.</span><span class="n">structure_perturbations</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="nb">tuple</span><span class="p">(</span><span class="n">best_edge</span><span class="p">))</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">feature_perturbations</span><span class="o">.</span><span class="n">append</span><span class="p">(())</span>
                <span class="n">surrogate_losses</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">best_edge_score</span><span class="p">)</span>

                <span class="c1"># Update likelihood ratio test values</span>
                <span class="n">current_S_d</span> <span class="o">=</span> <span class="n">new_S_d</span><span class="p">[</span><span class="n">powerlaw_filter</span><span class="p">][</span><span class="n">best_edge_ix</span><span class="p">]</span>
                <span class="n">current_n</span> <span class="o">=</span> <span class="n">new_n</span><span class="p">[</span><span class="n">powerlaw_filter</span><span class="p">][</span><span class="n">best_edge_ix</span><span class="p">]</span>
                <span class="n">current_degree_sequence</span><span class="p">[</span><span class="n">best_edge</span><span class="p">]</span> <span class="o">+=</span> <span class="n">deltas</span><span class="p">[</span><span class="n">powerlaw_filter</span><span class="p">][</span><span class="n">best_edge_ix</span><span class="p">]</span>

            <span class="k">else</span><span class="p">:</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="p">[</span><span class="nb">tuple</span><span class="p">(</span><span class="n">best_feature_ix</span><span class="p">)]</span> <span class="o">=</span> <span class="mi">1</span> <span class="o">-</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="p">[</span><span class="nb">tuple</span><span class="p">(</span><span class="n">best_feature_ix</span><span class="p">)]</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">feature_perturbations</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="nb">tuple</span><span class="p">(</span><span class="n">best_feature_ix</span><span class="p">))</span>
                <span class="bp">self</span><span class="o">.</span><span class="n">structure_perturbations</span><span class="o">.</span><span class="n">append</span><span class="p">(())</span>
                <span class="n">surrogate_losses</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">best_feature_score</span><span class="p">)</span></div>

        <span class="c1"># return self.modified_adj, self.modified_features</span>

<div class="viewcode-block" id="Nettack.get_attacker_nodes"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.Nettack.get_attacker_nodes">[docs]</a>    <span class="k">def</span> <span class="nf">get_attacker_nodes</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">n</span><span class="o">=</span><span class="mi">5</span><span class="p">,</span> <span class="n">add_additional_nodes</span> <span class="o">=</span> <span class="kc">False</span><span class="p">):</span>
        <span class="sd">&quot;&quot;&quot;Determine the influencer nodes to attack node i based on</span>
<span class="sd">        the weights W and the attributes X.</span>
<span class="sd">        &quot;&quot;&quot;</span>
        <span class="k">assert</span> <span class="n">n</span> <span class="o">&lt;</span> <span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="o">-</span><span class="mi">1</span><span class="p">,</span> <span class="s2">&quot;number of influencers cannot be &gt;= number of nodes in the graph!&quot;</span>
        <span class="n">neighbors</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">ori_adj</span><span class="p">[</span><span class="bp">self</span><span class="o">.</span><span class="n">target_node</span><span class="p">]</span><span class="o">.</span><span class="n">nonzero</span><span class="p">()[</span><span class="mi">1</span><span class="p">]</span>
        <span class="k">assert</span> <span class="bp">self</span><span class="o">.</span><span class="n">target_node</span> <span class="ow">not</span> <span class="ow">in</span> <span class="n">neighbors</span>

        <span class="n">potential_edges</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">column_stack</span><span class="p">((</span><span class="n">np</span><span class="o">.</span><span class="n">tile</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">target_node</span><span class="p">,</span> <span class="nb">len</span><span class="p">(</span><span class="n">neighbors</span><span class="p">)),</span><span class="n">neighbors</span><span class="p">))</span><span class="o">.</span><span class="n">astype</span><span class="p">(</span><span class="s2">&quot;int32&quot;</span><span class="p">)</span>

        <span class="c1"># The new A_hat_square_uv values that we would get if we removed the edge from u to each of the neighbors, respectively</span>
        <span class="n">a_hat_uv</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">compute_new_a_hat_uv</span><span class="p">(</span><span class="n">potential_edges</span><span class="p">,</span> <span class="bp">self</span><span class="o">.</span><span class="n">target_node</span><span class="p">)</span>

        <span class="c1"># XW = self.compute_XW()</span>
        <span class="n">XW</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span> <span class="o">@</span> <span class="bp">self</span><span class="o">.</span><span class="n">W</span>

        <span class="c1"># compute the struct scores for all neighbors</span>
        <span class="n">struct_scores</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">struct_score</span><span class="p">(</span><span class="n">a_hat_uv</span><span class="p">,</span> <span class="n">XW</span><span class="p">)</span>
        <span class="k">if</span> <span class="nb">len</span><span class="p">(</span><span class="n">neighbors</span><span class="p">)</span> <span class="o">&gt;=</span> <span class="n">n</span><span class="p">:</span>  <span class="c1"># do we have enough neighbors for the number of desired influencers?</span>
            <span class="n">influencer_nodes</span> <span class="o">=</span> <span class="n">neighbors</span><span class="p">[</span><span class="n">np</span><span class="o">.</span><span class="n">argsort</span><span class="p">(</span><span class="n">struct_scores</span><span class="p">)[:</span><span class="n">n</span><span class="p">]]</span>
            <span class="k">if</span> <span class="n">add_additional_nodes</span><span class="p">:</span>
                <span class="k">return</span> <span class="n">influencer_nodes</span><span class="p">,</span> <span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">([])</span>
            <span class="k">return</span> <span class="n">influencer_nodes</span>
        <span class="k">else</span><span class="p">:</span>

            <span class="n">influencer_nodes</span> <span class="o">=</span> <span class="n">neighbors</span>
            <span class="k">if</span> <span class="n">add_additional_nodes</span><span class="p">:</span>  <span class="c1"># Add additional influencers by connecting them to u first.</span>
                <span class="c1"># Compute the set of possible additional influencers, i.e. all nodes except the ones</span>
                <span class="c1"># that are already connected to u.</span>
                <span class="n">poss_add_infl</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">setdiff1d</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">setdiff1d</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">arange</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="p">),</span><span class="n">neighbors</span><span class="p">),</span> <span class="bp">self</span><span class="o">.</span><span class="n">target_node</span><span class="p">)</span>
                <span class="n">n_possible_additional</span> <span class="o">=</span> <span class="nb">len</span><span class="p">(</span><span class="n">poss_add_infl</span><span class="p">)</span>
                <span class="n">n_additional_attackers</span> <span class="o">=</span> <span class="n">n</span><span class="o">-</span><span class="nb">len</span><span class="p">(</span><span class="n">neighbors</span><span class="p">)</span>
                <span class="n">possible_edges</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">column_stack</span><span class="p">((</span><span class="n">np</span><span class="o">.</span><span class="n">tile</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">target_node</span><span class="p">,</span> <span class="n">n_possible_additional</span><span class="p">),</span> <span class="n">poss_add_infl</span><span class="p">))</span>

                <span class="c1"># Compute the struct_scores for all possible additional influencers, and choose the one</span>
                <span class="c1"># with the best struct score.</span>
                <span class="n">a_hat_uv_additional</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">compute_new_a_hat_uv</span><span class="p">(</span><span class="n">possible_edges</span><span class="p">,</span> <span class="bp">self</span><span class="o">.</span><span class="n">target_node</span><span class="p">)</span>
                <span class="n">additional_struct_scores</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">struct_score</span><span class="p">(</span><span class="n">a_hat_uv_additional</span><span class="p">,</span> <span class="n">XW</span><span class="p">)</span>
                <span class="n">additional_influencers</span> <span class="o">=</span> <span class="n">poss_add_infl</span><span class="p">[</span><span class="n">np</span><span class="o">.</span><span class="n">argsort</span><span class="p">(</span><span class="n">additional_struct_scores</span><span class="p">)[</span><span class="o">-</span><span class="n">n_additional_attackers</span><span class="p">::]]</span>

                <span class="k">return</span> <span class="n">influencer_nodes</span><span class="p">,</span> <span class="n">additional_influencers</span>
            <span class="k">else</span><span class="p">:</span>
                <span class="k">return</span> <span class="n">influencer_nodes</span></div>

    <span class="k">def</span> <span class="nf">compute_logits</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
        <span class="k">return</span> <span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span> <span class="o">@</span> <span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span> <span class="o">@</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span> <span class="o">@</span> <span class="bp">self</span><span class="o">.</span><span class="n">W</span><span class="p">)[</span><span class="bp">self</span><span class="o">.</span><span class="n">target_node</span><span class="p">]</span>

    <span class="k">def</span> <span class="nf">strongest_wrong_class</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">logits</span><span class="p">):</span>
        <span class="n">label_u_onehot</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">eye</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">nclass</span><span class="p">)[</span><span class="bp">self</span><span class="o">.</span><span class="n">label_u</span><span class="p">]</span>
        <span class="k">return</span> <span class="p">(</span><span class="n">logits</span> <span class="o">-</span> <span class="mi">1000</span><span class="o">*</span><span class="n">label_u_onehot</span><span class="p">)</span><span class="o">.</span><span class="n">argmax</span><span class="p">()</span>

<div class="viewcode-block" id="Nettack.feature_scores"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.Nettack.feature_scores">[docs]</a>    <span class="k">def</span> <span class="nf">feature_scores</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
        <span class="sd">&quot;&quot;&quot;Compute feature scores for all possible feature changes.</span>
<span class="sd">        &quot;&quot;&quot;</span>

        <span class="k">if</span> <span class="bp">self</span><span class="o">.</span><span class="n">cooc_constraint</span> <span class="ow">is</span> <span class="kc">None</span><span class="p">:</span>
            <span class="bp">self</span><span class="o">.</span><span class="n">compute_cooccurrence_constraint</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span><span class="p">)</span>
        <span class="n">logits</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">compute_logits</span><span class="p">()</span>
        <span class="n">best_wrong_class</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">strongest_wrong_class</span><span class="p">(</span><span class="n">logits</span><span class="p">)</span>
        <span class="n">surrogate_loss</span> <span class="o">=</span> <span class="n">logits</span><span class="p">[</span><span class="bp">self</span><span class="o">.</span><span class="n">label_u</span><span class="p">]</span> <span class="o">-</span> <span class="n">logits</span><span class="p">[</span><span class="n">best_wrong_class</span><span class="p">]</span>

        <span class="n">gradient</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">gradient_wrt_x</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">label_u</span><span class="p">)</span> <span class="o">-</span> <span class="bp">self</span><span class="o">.</span><span class="n">gradient_wrt_x</span><span class="p">(</span><span class="n">best_wrong_class</span><span class="p">)</span>
        <span class="c1"># gradients_flipped = (gradient * -1).tolil()</span>
        <span class="n">gradients_flipped</span> <span class="o">=</span> <span class="n">sp</span><span class="o">.</span><span class="n">lil_matrix</span><span class="p">(</span><span class="n">gradient</span> <span class="o">*</span> <span class="o">-</span><span class="mi">1</span><span class="p">)</span>
        <span class="n">gradients_flipped</span><span class="p">[</span><span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="o">.</span><span class="n">nonzero</span><span class="p">()]</span> <span class="o">*=</span> <span class="o">-</span><span class="mi">1</span>

        <span class="n">X_influencers</span> <span class="o">=</span> <span class="n">sp</span><span class="o">.</span><span class="n">lil_matrix</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="o">.</span><span class="n">shape</span><span class="p">)</span>
        <span class="n">X_influencers</span><span class="p">[</span><span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span><span class="p">]</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="p">[</span><span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span><span class="p">]</span>
        <span class="n">gradients_flipped</span> <span class="o">=</span> <span class="n">gradients_flipped</span><span class="o">.</span><span class="n">multiply</span><span class="p">((</span><span class="bp">self</span><span class="o">.</span><span class="n">cooc_constraint</span> <span class="o">+</span> <span class="n">X_influencers</span><span class="p">)</span> <span class="o">&gt;</span> <span class="mi">0</span><span class="p">)</span>
        <span class="n">nnz_ixs</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">(</span><span class="n">gradients_flipped</span><span class="o">.</span><span class="n">nonzero</span><span class="p">())</span><span class="o">.</span><span class="n">T</span>

        <span class="n">sorting</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">argsort</span><span class="p">(</span><span class="n">gradients_flipped</span><span class="p">[</span><span class="nb">tuple</span><span class="p">(</span><span class="n">nnz_ixs</span><span class="o">.</span><span class="n">T</span><span class="p">)])</span><span class="o">.</span><span class="n">A1</span>
        <span class="n">sorted_ixs</span> <span class="o">=</span> <span class="n">nnz_ixs</span><span class="p">[</span><span class="n">sorting</span><span class="p">]</span>
        <span class="n">grads</span> <span class="o">=</span> <span class="n">gradients_flipped</span><span class="p">[</span><span class="nb">tuple</span><span class="p">(</span><span class="n">nnz_ixs</span><span class="p">[</span><span class="n">sorting</span><span class="p">]</span><span class="o">.</span><span class="n">T</span><span class="p">)]</span>

        <span class="n">scores</span> <span class="o">=</span> <span class="n">surrogate_loss</span> <span class="o">-</span> <span class="n">grads</span>
        <span class="k">return</span> <span class="n">sorted_ixs</span><span class="p">[::</span><span class="o">-</span><span class="mi">1</span><span class="p">],</span> <span class="n">scores</span><span class="o">.</span><span class="n">A1</span><span class="p">[::</span><span class="o">-</span><span class="mi">1</span><span class="p">]</span></div>

<div class="viewcode-block" id="Nettack.compute_cooccurrence_constraint"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.Nettack.compute_cooccurrence_constraint">[docs]</a>    <span class="k">def</span> <span class="nf">compute_cooccurrence_constraint</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">nodes</span><span class="p">):</span>
        <span class="sd">&quot;&quot;&quot;</span>
<span class="sd">        Co-occurrence constraint as described in the paper.</span>

<span class="sd">        Parameters</span>
<span class="sd">        ----------</span>
<span class="sd">        nodes: np.array</span>
<span class="sd">            Nodes whose features are considered for change</span>

<span class="sd">        Returns</span>
<span class="sd">        -------</span>
<span class="sd">        np.array [len(nodes), D], dtype bool</span>
<span class="sd">            Binary matrix of dimension len(nodes) x D. A 1 in entry n,d indicates that</span>
<span class="sd">            we are allowed to add feature d to the features of node n.</span>

<span class="sd">        &quot;&quot;&quot;</span>

        <span class="n">words_graph</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">cooc_matrix</span><span class="o">.</span><span class="n">copy</span><span class="p">()</span>
        <span class="n">D</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="o">.</span><span class="n">shape</span><span class="p">[</span><span class="mi">1</span><span class="p">]</span>
        <span class="n">words_graph</span><span class="o">.</span><span class="n">setdiag</span><span class="p">(</span><span class="mi">0</span><span class="p">)</span>
        <span class="n">words_graph</span> <span class="o">=</span> <span class="p">(</span><span class="n">words_graph</span> <span class="o">&gt;</span> <span class="mi">0</span><span class="p">)</span>
        <span class="n">word_degrees</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="n">words_graph</span><span class="p">,</span> <span class="n">axis</span><span class="o">=</span><span class="mi">0</span><span class="p">)</span><span class="o">.</span><span class="n">A1</span>

        <span class="n">inv_word_degrees</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">reciprocal</span><span class="p">(</span><span class="n">word_degrees</span><span class="o">.</span><span class="n">astype</span><span class="p">(</span><span class="nb">float</span><span class="p">)</span> <span class="o">+</span> <span class="mf">1e-8</span><span class="p">)</span>

        <span class="n">sd</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">zeros</span><span class="p">([</span><span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="p">])</span>
        <span class="k">for</span> <span class="n">n</span> <span class="ow">in</span> <span class="nb">range</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="p">):</span>
            <span class="n">n_idx</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="p">[</span><span class="n">n</span><span class="p">,</span> <span class="p">:]</span><span class="o">.</span><span class="n">nonzero</span><span class="p">()[</span><span class="mi">1</span><span class="p">]</span>
            <span class="n">sd</span><span class="p">[</span><span class="n">n</span><span class="p">]</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="n">inv_word_degrees</span><span class="p">[</span><span class="n">n_idx</span><span class="o">.</span><span class="n">tolist</span><span class="p">()])</span>

        <span class="n">scores_matrix</span> <span class="o">=</span> <span class="n">sp</span><span class="o">.</span><span class="n">lil_matrix</span><span class="p">((</span><span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="p">,</span> <span class="n">D</span><span class="p">))</span>

        <span class="k">for</span> <span class="n">n</span> <span class="ow">in</span> <span class="n">nodes</span><span class="p">:</span>
            <span class="n">common_words</span> <span class="o">=</span> <span class="n">words_graph</span><span class="o">.</span><span class="n">multiply</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span><span class="p">[</span><span class="n">n</span><span class="p">])</span>
            <span class="n">idegs</span> <span class="o">=</span> <span class="n">inv_word_degrees</span><span class="p">[</span><span class="n">common_words</span><span class="o">.</span><span class="n">nonzero</span><span class="p">()[</span><span class="mi">1</span><span class="p">]]</span>
            <span class="n">nnz</span> <span class="o">=</span> <span class="n">common_words</span><span class="o">.</span><span class="n">nonzero</span><span class="p">()[</span><span class="mi">0</span><span class="p">]</span>
            <span class="n">scores</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">([</span><span class="n">idegs</span><span class="p">[</span><span class="n">nnz</span> <span class="o">==</span> <span class="n">ix</span><span class="p">]</span><span class="o">.</span><span class="n">sum</span><span class="p">()</span> <span class="k">for</span> <span class="n">ix</span> <span class="ow">in</span> <span class="nb">range</span><span class="p">(</span><span class="n">D</span><span class="p">)])</span>
            <span class="n">scores_matrix</span><span class="p">[</span><span class="n">n</span><span class="p">]</span> <span class="o">=</span> <span class="n">scores</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">cooc_constraint</span> <span class="o">=</span> <span class="n">sp</span><span class="o">.</span><span class="n">csr_matrix</span><span class="p">(</span><span class="n">scores_matrix</span> <span class="o">-</span> <span class="mf">0.5</span> <span class="o">*</span> <span class="n">sd</span><span class="p">[:,</span> <span class="kc">None</span><span class="p">]</span> <span class="o">&gt;</span> <span class="mi">0</span><span class="p">)</span></div>

    <span class="k">def</span> <span class="nf">gradient_wrt_x</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">label</span><span class="p">):</span>
        <span class="c1"># return self.adj_norm.dot(self.adj_norm)[self.target_node].T.dot(self.W[:, label].T)</span>
        <span class="k">return</span> <span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span><span class="o">.</span><span class="n">dot</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span><span class="p">)[</span><span class="bp">self</span><span class="o">.</span><span class="n">target_node</span><span class="p">]</span><span class="o">.</span><span class="n">T</span><span class="o">.</span><span class="n">dot</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">W</span><span class="p">[:,</span> <span class="n">label</span><span class="p">]</span><span class="o">.</span><span class="n">reshape</span><span class="p">(</span><span class="mi">1</span><span class="p">,</span> <span class="o">-</span><span class="mi">1</span><span class="p">))</span>

<div class="viewcode-block" id="Nettack.reset"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.Nettack.reset">[docs]</a>    <span class="k">def</span> <span class="nf">reset</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
        <span class="sd">&quot;&quot;&quot;Reset Nettack</span>
<span class="sd">        &quot;&quot;&quot;</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">ori_adj</span><span class="o">.</span><span class="n">copy</span><span class="p">()</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">modified_features</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">ori_features</span><span class="o">.</span><span class="n">copy</span><span class="p">()</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">structure_perturbations</span> <span class="o">=</span> <span class="p">[]</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">feature_perturbations</span> <span class="o">=</span> <span class="p">[]</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">influencer_nodes</span> <span class="o">=</span> <span class="p">[]</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">potential_edges</span> <span class="o">=</span> <span class="p">[]</span>
        <span class="bp">self</span><span class="o">.</span><span class="n">cooc_constraint</span> <span class="o">=</span> <span class="kc">None</span></div>


<div class="viewcode-block" id="Nettack.struct_score"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.Nettack.struct_score">[docs]</a>    <span class="k">def</span> <span class="nf">struct_score</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">a_hat_uv</span><span class="p">,</span> <span class="n">XW</span><span class="p">):</span>
        <span class="sd">&quot;&quot;&quot;</span>
<span class="sd">        Compute structure scores, cf. Eq. 15 in the paper</span>

<span class="sd">        Parameters</span>
<span class="sd">        ----------</span>
<span class="sd">        a_hat_uv: sp.sparse_matrix, shape [P,2]</span>
<span class="sd">            Entries of matrix A_hat^2_u for each potential edge (see paper for explanation)</span>

<span class="sd">        XW: sp.sparse_matrix, shape [N, K], dtype float</span>
<span class="sd">            The class logits for each node.</span>

<span class="sd">        Returns</span>
<span class="sd">        -------</span>
<span class="sd">        np.array [P,]</span>
<span class="sd">            The struct score for every row in a_hat_uv</span>
<span class="sd">        &quot;&quot;&quot;</span>

        <span class="n">logits</span> <span class="o">=</span> <span class="n">a_hat_uv</span><span class="o">.</span><span class="n">dot</span><span class="p">(</span><span class="n">XW</span><span class="p">)</span>
        <span class="n">label_onehot</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">eye</span><span class="p">(</span><span class="n">XW</span><span class="o">.</span><span class="n">shape</span><span class="p">[</span><span class="mi">1</span><span class="p">])[</span><span class="bp">self</span><span class="o">.</span><span class="n">label_u</span><span class="p">]</span>
        <span class="n">best_wrong_class_logits</span> <span class="o">=</span> <span class="p">(</span><span class="n">logits</span> <span class="o">-</span> <span class="mi">1000</span> <span class="o">*</span> <span class="n">label_onehot</span><span class="p">)</span><span class="o">.</span><span class="n">max</span><span class="p">(</span><span class="mi">1</span><span class="p">)</span>
        <span class="n">logits_for_correct_class</span> <span class="o">=</span> <span class="n">logits</span><span class="p">[:,</span><span class="bp">self</span><span class="o">.</span><span class="n">label_u</span><span class="p">]</span>
        <span class="n">struct_scores</span> <span class="o">=</span> <span class="n">logits_for_correct_class</span> <span class="o">-</span> <span class="n">best_wrong_class_logits</span>

        <span class="k">return</span> <span class="n">struct_scores</span></div>

<div class="viewcode-block" id="Nettack.compute_new_a_hat_uv"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.Nettack.compute_new_a_hat_uv">[docs]</a>    <span class="k">def</span> <span class="nf">compute_new_a_hat_uv</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">potential_edges</span><span class="p">,</span> <span class="n">target_node</span><span class="p">):</span>
        <span class="sd">&quot;&quot;&quot;</span>
<span class="sd">        Compute the updated A_hat_square_uv entries that would result from inserting/deleting the input edges,</span>
<span class="sd">        for every edge.</span>

<span class="sd">        Parameters</span>
<span class="sd">        ----------</span>
<span class="sd">        potential_edges: np.array, shape [P,2], dtype int</span>
<span class="sd">            The edges to check.</span>

<span class="sd">        Returns</span>
<span class="sd">        -------</span>
<span class="sd">        sp.sparse_matrix: updated A_hat_square_u entries, a sparse PxN matrix, where P is len(possible_edges).</span>
<span class="sd">        &quot;&quot;&quot;</span>

        <span class="n">edges</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">(</span><span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="o">.</span><span class="n">nonzero</span><span class="p">())</span><span class="o">.</span><span class="n">T</span>
        <span class="n">edges_set</span> <span class="o">=</span> <span class="p">{</span><span class="nb">tuple</span><span class="p">(</span><span class="n">x</span><span class="p">)</span> <span class="k">for</span> <span class="n">x</span> <span class="ow">in</span> <span class="n">edges</span><span class="p">}</span>
        <span class="n">A_hat_sq</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span> <span class="o">@</span> <span class="bp">self</span><span class="o">.</span><span class="n">adj_norm</span>
        <span class="n">values_before</span> <span class="o">=</span> <span class="n">A_hat_sq</span><span class="p">[</span><span class="n">target_node</span><span class="p">]</span><span class="o">.</span><span class="n">toarray</span><span class="p">()[</span><span class="mi">0</span><span class="p">]</span>
        <span class="n">node_ixs</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">unique</span><span class="p">(</span><span class="n">edges</span><span class="p">[:,</span> <span class="mi">0</span><span class="p">],</span> <span class="n">return_index</span><span class="o">=</span><span class="kc">True</span><span class="p">)[</span><span class="mi">1</span><span class="p">]</span>
        <span class="n">twohop_ixs</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">(</span><span class="n">A_hat_sq</span><span class="o">.</span><span class="n">nonzero</span><span class="p">())</span><span class="o">.</span><span class="n">T</span>
        <span class="n">degrees</span> <span class="o">=</span> <span class="bp">self</span><span class="o">.</span><span class="n">modified_adj</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="mi">0</span><span class="p">)</span><span class="o">.</span><span class="n">A1</span> <span class="o">+</span> <span class="mi">1</span>

        <span class="n">ixs</span><span class="p">,</span> <span class="n">vals</span> <span class="o">=</span> <span class="n">compute_new_a_hat_uv</span><span class="p">(</span><span class="n">edges</span><span class="p">,</span> <span class="n">node_ixs</span><span class="p">,</span> <span class="n">edges_set</span><span class="p">,</span> <span class="n">twohop_ixs</span><span class="p">,</span> <span class="n">values_before</span><span class="p">,</span> <span class="n">degrees</span><span class="p">,</span>
                                         <span class="n">potential_edges</span><span class="o">.</span><span class="n">astype</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">int32</span><span class="p">),</span> <span class="n">target_node</span><span class="p">)</span>
        <span class="n">ixs_arr</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">(</span><span class="n">ixs</span><span class="p">)</span>
        <span class="n">a_hat_uv</span> <span class="o">=</span> <span class="n">sp</span><span class="o">.</span><span class="n">coo_matrix</span><span class="p">((</span><span class="n">vals</span><span class="p">,</span> <span class="p">(</span><span class="n">ixs_arr</span><span class="p">[:,</span> <span class="mi">0</span><span class="p">],</span> <span class="n">ixs_arr</span><span class="p">[:,</span> <span class="mi">1</span><span class="p">])),</span> <span class="n">shape</span><span class="o">=</span><span class="p">[</span><span class="nb">len</span><span class="p">(</span><span class="n">potential_edges</span><span class="p">),</span> <span class="bp">self</span><span class="o">.</span><span class="n">nnodes</span><span class="p">])</span>

        <span class="k">return</span> <span class="n">a_hat_uv</span></div></div>

<span class="nd">@jit</span><span class="p">(</span><span class="n">nopython</span><span class="o">=</span><span class="kc">True</span><span class="p">)</span>
<span class="k">def</span> <span class="nf">connected_after</span><span class="p">(</span><span class="n">u</span><span class="p">,</span> <span class="n">v</span><span class="p">,</span> <span class="n">connected_before</span><span class="p">,</span> <span class="n">delta</span><span class="p">):</span>
    <span class="k">if</span> <span class="n">u</span> <span class="o">==</span> <span class="n">v</span><span class="p">:</span>
        <span class="k">if</span> <span class="n">delta</span> <span class="o">==</span> <span class="o">-</span><span class="mi">1</span><span class="p">:</span>
            <span class="k">return</span> <span class="kc">False</span>
        <span class="k">else</span><span class="p">:</span>
            <span class="k">return</span> <span class="kc">True</span>
    <span class="k">else</span><span class="p">:</span>
        <span class="k">return</span> <span class="n">connected_before</span>


<div class="viewcode-block" id="compute_new_a_hat_uv"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.compute_new_a_hat_uv">[docs]</a><span class="nd">@jit</span><span class="p">(</span><span class="n">nopython</span><span class="o">=</span><span class="kc">True</span><span class="p">)</span>
<span class="k">def</span> <span class="nf">compute_new_a_hat_uv</span><span class="p">(</span><span class="n">edge_ixs</span><span class="p">,</span> <span class="n">node_nb_ixs</span><span class="p">,</span> <span class="n">edges_set</span><span class="p">,</span> <span class="n">twohop_ixs</span><span class="p">,</span> <span class="n">values_before</span><span class="p">,</span> <span class="n">degs</span><span class="p">,</span> <span class="n">potential_edges</span><span class="p">,</span> <span class="n">u</span><span class="p">):</span>
    <span class="sd">&quot;&quot;&quot;</span>
<span class="sd">    Compute the new values [A_hat_square]_u for every potential edge, where u is the target node. C.f. Theorem 5.1</span>
<span class="sd">    equation 17.</span>

<span class="sd">    &quot;&quot;&quot;</span>
    <span class="n">N</span> <span class="o">=</span> <span class="n">degs</span><span class="o">.</span><span class="n">shape</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span>

    <span class="n">twohop_u</span> <span class="o">=</span> <span class="n">twohop_ixs</span><span class="p">[</span><span class="n">twohop_ixs</span><span class="p">[:,</span> <span class="mi">0</span><span class="p">]</span> <span class="o">==</span> <span class="n">u</span><span class="p">,</span> <span class="mi">1</span><span class="p">]</span>
    <span class="n">nbs_u</span> <span class="o">=</span> <span class="n">edge_ixs</span><span class="p">[</span><span class="n">edge_ixs</span><span class="p">[:,</span> <span class="mi">0</span><span class="p">]</span> <span class="o">==</span> <span class="n">u</span><span class="p">,</span> <span class="mi">1</span><span class="p">]</span>
    <span class="n">nbs_u_set</span> <span class="o">=</span> <span class="nb">set</span><span class="p">(</span><span class="n">nbs_u</span><span class="p">)</span>

    <span class="n">return_ixs</span> <span class="o">=</span> <span class="p">[]</span>
    <span class="n">return_values</span> <span class="o">=</span> <span class="p">[]</span>

    <span class="k">for</span> <span class="n">ix</span> <span class="ow">in</span> <span class="nb">range</span><span class="p">(</span><span class="nb">len</span><span class="p">(</span><span class="n">potential_edges</span><span class="p">)):</span>
        <span class="n">edge</span> <span class="o">=</span> <span class="n">potential_edges</span><span class="p">[</span><span class="n">ix</span><span class="p">]</span>
        <span class="n">edge_set</span> <span class="o">=</span> <span class="nb">set</span><span class="p">(</span><span class="n">edge</span><span class="p">)</span>
        <span class="n">degs_new</span> <span class="o">=</span> <span class="n">degs</span><span class="o">.</span><span class="n">copy</span><span class="p">()</span>
        <span class="n">delta</span> <span class="o">=</span> <span class="o">-</span><span class="mi">2</span> <span class="o">*</span> <span class="p">((</span><span class="n">edge</span><span class="p">[</span><span class="mi">0</span><span class="p">],</span> <span class="n">edge</span><span class="p">[</span><span class="mi">1</span><span class="p">])</span> <span class="ow">in</span> <span class="n">edges_set</span><span class="p">)</span> <span class="o">+</span> <span class="mi">1</span>
        <span class="n">degs_new</span><span class="p">[</span><span class="n">edge</span><span class="p">]</span> <span class="o">+=</span> <span class="n">delta</span>

        <span class="n">nbs_edge0</span> <span class="o">=</span> <span class="n">edge_ixs</span><span class="p">[</span><span class="n">edge_ixs</span><span class="p">[:,</span> <span class="mi">0</span><span class="p">]</span> <span class="o">==</span> <span class="n">edge</span><span class="p">[</span><span class="mi">0</span><span class="p">],</span> <span class="mi">1</span><span class="p">]</span>
        <span class="n">nbs_edge1</span> <span class="o">=</span> <span class="n">edge_ixs</span><span class="p">[</span><span class="n">edge_ixs</span><span class="p">[:,</span> <span class="mi">0</span><span class="p">]</span> <span class="o">==</span> <span class="n">edge</span><span class="p">[</span><span class="mi">1</span><span class="p">],</span> <span class="mi">1</span><span class="p">]</span>

        <span class="n">affected_nodes</span> <span class="o">=</span> <span class="nb">set</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">concatenate</span><span class="p">((</span><span class="n">twohop_u</span><span class="p">,</span> <span class="n">nbs_edge0</span><span class="p">,</span> <span class="n">nbs_edge1</span><span class="p">)))</span>
        <span class="n">affected_nodes</span> <span class="o">=</span> <span class="n">affected_nodes</span><span class="o">.</span><span class="n">union</span><span class="p">(</span><span class="n">edge_set</span><span class="p">)</span>
        <span class="n">a_um</span> <span class="o">=</span> <span class="n">edge</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span> <span class="ow">in</span> <span class="n">nbs_u_set</span>
        <span class="n">a_un</span> <span class="o">=</span> <span class="n">edge</span><span class="p">[</span><span class="mi">1</span><span class="p">]</span> <span class="ow">in</span> <span class="n">nbs_u_set</span>

        <span class="n">a_un_after</span> <span class="o">=</span> <span class="n">connected_after</span><span class="p">(</span><span class="n">u</span><span class="p">,</span> <span class="n">edge</span><span class="p">[</span><span class="mi">0</span><span class="p">],</span> <span class="n">a_un</span><span class="p">,</span> <span class="n">delta</span><span class="p">)</span>
        <span class="n">a_um_after</span> <span class="o">=</span> <span class="n">connected_after</span><span class="p">(</span><span class="n">u</span><span class="p">,</span> <span class="n">edge</span><span class="p">[</span><span class="mi">1</span><span class="p">],</span> <span class="n">a_um</span><span class="p">,</span> <span class="n">delta</span><span class="p">)</span>

        <span class="k">for</span> <span class="n">v</span> <span class="ow">in</span> <span class="n">affected_nodes</span><span class="p">:</span>
            <span class="n">a_uv_before</span> <span class="o">=</span> <span class="n">v</span> <span class="ow">in</span> <span class="n">nbs_u_set</span>
            <span class="n">a_uv_before_sl</span> <span class="o">=</span> <span class="n">a_uv_before</span> <span class="ow">or</span> <span class="n">v</span> <span class="o">==</span> <span class="n">u</span>

            <span class="k">if</span> <span class="n">v</span> <span class="ow">in</span> <span class="n">edge_set</span> <span class="ow">and</span> <span class="n">u</span> <span class="ow">in</span> <span class="n">edge_set</span> <span class="ow">and</span> <span class="n">u</span> <span class="o">!=</span> <span class="n">v</span><span class="p">:</span>
                <span class="k">if</span> <span class="n">delta</span> <span class="o">==</span> <span class="o">-</span><span class="mi">1</span><span class="p">:</span>
                    <span class="n">a_uv_after</span> <span class="o">=</span> <span class="kc">False</span>
                <span class="k">else</span><span class="p">:</span>
                    <span class="n">a_uv_after</span> <span class="o">=</span> <span class="kc">True</span>
            <span class="k">else</span><span class="p">:</span>
                <span class="n">a_uv_after</span> <span class="o">=</span> <span class="n">a_uv_before</span>
            <span class="n">a_uv_after_sl</span> <span class="o">=</span> <span class="n">a_uv_after</span> <span class="ow">or</span> <span class="n">v</span> <span class="o">==</span> <span class="n">u</span>

            <span class="n">from_ix</span> <span class="o">=</span> <span class="n">node_nb_ixs</span><span class="p">[</span><span class="n">v</span><span class="p">]</span>
            <span class="n">to_ix</span> <span class="o">=</span> <span class="n">node_nb_ixs</span><span class="p">[</span><span class="n">v</span> <span class="o">+</span> <span class="mi">1</span><span class="p">]</span> <span class="k">if</span> <span class="n">v</span> <span class="o">&lt;</span> <span class="n">N</span> <span class="o">-</span> <span class="mi">1</span> <span class="k">else</span> <span class="nb">len</span><span class="p">(</span><span class="n">edge_ixs</span><span class="p">)</span>
            <span class="n">node_nbs</span> <span class="o">=</span> <span class="n">edge_ixs</span><span class="p">[</span><span class="n">from_ix</span><span class="p">:</span><span class="n">to_ix</span><span class="p">,</span> <span class="mi">1</span><span class="p">]</span>
            <span class="n">node_nbs_set</span> <span class="o">=</span> <span class="nb">set</span><span class="p">(</span><span class="n">node_nbs</span><span class="p">)</span>
            <span class="n">a_vm_before</span> <span class="o">=</span> <span class="n">edge</span><span class="p">[</span><span class="mi">0</span><span class="p">]</span> <span class="ow">in</span> <span class="n">node_nbs_set</span>

            <span class="n">a_vn_before</span> <span class="o">=</span> <span class="n">edge</span><span class="p">[</span><span class="mi">1</span><span class="p">]</span> <span class="ow">in</span> <span class="n">node_nbs_set</span>
            <span class="n">a_vn_after</span> <span class="o">=</span> <span class="n">connected_after</span><span class="p">(</span><span class="n">v</span><span class="p">,</span> <span class="n">edge</span><span class="p">[</span><span class="mi">0</span><span class="p">],</span> <span class="n">a_vn_before</span><span class="p">,</span> <span class="n">delta</span><span class="p">)</span>
            <span class="n">a_vm_after</span> <span class="o">=</span> <span class="n">connected_after</span><span class="p">(</span><span class="n">v</span><span class="p">,</span> <span class="n">edge</span><span class="p">[</span><span class="mi">1</span><span class="p">],</span> <span class="n">a_vm_before</span><span class="p">,</span> <span class="n">delta</span><span class="p">)</span>

            <span class="n">mult_term</span> <span class="o">=</span> <span class="mi">1</span> <span class="o">/</span> <span class="n">np</span><span class="o">.</span><span class="n">sqrt</span><span class="p">(</span><span class="n">degs_new</span><span class="p">[</span><span class="n">u</span><span class="p">]</span> <span class="o">*</span> <span class="n">degs_new</span><span class="p">[</span><span class="n">v</span><span class="p">])</span>

            <span class="n">sum_term1</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">sqrt</span><span class="p">(</span><span class="n">degs</span><span class="p">[</span><span class="n">u</span><span class="p">]</span> <span class="o">*</span> <span class="n">degs</span><span class="p">[</span><span class="n">v</span><span class="p">])</span> <span class="o">*</span> <span class="n">values_before</span><span class="p">[</span><span class="n">v</span><span class="p">]</span> <span class="o">-</span> <span class="n">a_uv_before_sl</span> <span class="o">/</span> <span class="n">degs</span><span class="p">[</span><span class="n">u</span><span class="p">]</span> <span class="o">-</span> <span class="n">a_uv_before</span> <span class="o">/</span> \
                        <span class="n">degs</span><span class="p">[</span><span class="n">v</span><span class="p">]</span>
            <span class="n">sum_term2</span> <span class="o">=</span> <span class="n">a_uv_after</span> <span class="o">/</span> <span class="n">degs_new</span><span class="p">[</span><span class="n">v</span><span class="p">]</span> <span class="o">+</span> <span class="n">a_uv_after_sl</span> <span class="o">/</span> <span class="n">degs_new</span><span class="p">[</span><span class="n">u</span><span class="p">]</span>
            <span class="n">sum_term3</span> <span class="o">=</span> <span class="o">-</span><span class="p">((</span><span class="n">a_um</span> <span class="ow">and</span> <span class="n">a_vm_before</span><span class="p">)</span> <span class="o">/</span> <span class="n">degs</span><span class="p">[</span><span class="n">edge</span><span class="p">[</span><span class="mi">0</span><span class="p">]])</span> <span class="o">+</span> <span class="p">(</span><span class="n">a_um_after</span> <span class="ow">and</span> <span class="n">a_vm_after</span><span class="p">)</span> <span class="o">/</span> <span class="n">degs_new</span><span class="p">[</span><span class="n">edge</span><span class="p">[</span><span class="mi">0</span><span class="p">]]</span>
            <span class="n">sum_term4</span> <span class="o">=</span> <span class="o">-</span><span class="p">((</span><span class="n">a_un</span> <span class="ow">and</span> <span class="n">a_vn_before</span><span class="p">)</span> <span class="o">/</span> <span class="n">degs</span><span class="p">[</span><span class="n">edge</span><span class="p">[</span><span class="mi">1</span><span class="p">]])</span> <span class="o">+</span> <span class="p">(</span><span class="n">a_un_after</span> <span class="ow">and</span> <span class="n">a_vn_after</span><span class="p">)</span> <span class="o">/</span> <span class="n">degs_new</span><span class="p">[</span><span class="n">edge</span><span class="p">[</span><span class="mi">1</span><span class="p">]]</span>
            <span class="n">new_val</span> <span class="o">=</span> <span class="n">mult_term</span> <span class="o">*</span> <span class="p">(</span><span class="n">sum_term1</span> <span class="o">+</span> <span class="n">sum_term2</span> <span class="o">+</span> <span class="n">sum_term3</span> <span class="o">+</span> <span class="n">sum_term4</span><span class="p">)</span>

            <span class="n">return_ixs</span><span class="o">.</span><span class="n">append</span><span class="p">((</span><span class="n">ix</span><span class="p">,</span> <span class="n">v</span><span class="p">))</span>
            <span class="n">return_values</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">new_val</span><span class="p">)</span>

    <span class="k">return</span> <span class="n">return_ixs</span><span class="p">,</span> <span class="n">return_values</span></div>

<div class="viewcode-block" id="filter_singletons"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.filter_singletons">[docs]</a><span class="k">def</span> <span class="nf">filter_singletons</span><span class="p">(</span><span class="n">edges</span><span class="p">,</span> <span class="n">adj</span><span class="p">):</span>
    <span class="sd">&quot;&quot;&quot;</span>
<span class="sd">    Filter edges that, if removed, would turn one or more nodes into singleton nodes.</span>
<span class="sd">    &quot;&quot;&quot;</span>


    <span class="n">degs</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">squeeze</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="n">adj</span><span class="p">,</span><span class="mi">0</span><span class="p">)))</span>
    <span class="n">existing_edges</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">squeeze</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">(</span><span class="n">adj</span><span class="o">.</span><span class="n">tocsr</span><span class="p">()[</span><span class="nb">tuple</span><span class="p">(</span><span class="n">edges</span><span class="o">.</span><span class="n">T</span><span class="p">)]))</span>
    <span class="k">if</span> <span class="n">existing_edges</span><span class="o">.</span><span class="n">size</span> <span class="o">&gt;</span> <span class="mi">0</span><span class="p">:</span>
        <span class="n">edge_degrees</span> <span class="o">=</span> <span class="n">degs</span><span class="p">[</span><span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">(</span><span class="n">edges</span><span class="p">)]</span> <span class="o">+</span> <span class="mi">2</span><span class="o">*</span><span class="p">(</span><span class="mi">1</span><span class="o">-</span><span class="n">existing_edges</span><span class="p">[:,</span><span class="kc">None</span><span class="p">])</span> <span class="o">-</span> <span class="mi">1</span>
    <span class="k">else</span><span class="p">:</span>
        <span class="n">edge_degrees</span> <span class="o">=</span> <span class="n">degs</span><span class="p">[</span><span class="n">np</span><span class="o">.</span><span class="n">array</span><span class="p">(</span><span class="n">edges</span><span class="p">)]</span> <span class="o">+</span> <span class="mi">1</span>

    <span class="n">zeros</span> <span class="o">=</span> <span class="n">edge_degrees</span> <span class="o">==</span> <span class="mi">0</span>
    <span class="n">zeros_sum</span> <span class="o">=</span> <span class="n">zeros</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="mi">1</span><span class="p">)</span>
    <span class="k">return</span> <span class="n">zeros_sum</span> <span class="o">==</span> <span class="mi">0</span></div>

<div class="viewcode-block" id="compute_alpha"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.compute_alpha">[docs]</a><span class="k">def</span> <span class="nf">compute_alpha</span><span class="p">(</span><span class="n">n</span><span class="p">,</span> <span class="n">S_d</span><span class="p">,</span> <span class="n">d_min</span><span class="p">):</span>
    <span class="sd">&quot;&quot;&quot;</span>
<span class="sd">    Approximate the alpha of a power law distribution.</span>

<span class="sd">    &quot;&quot;&quot;</span>

    <span class="k">return</span> <span class="n">n</span> <span class="o">/</span> <span class="p">(</span><span class="n">S_d</span> <span class="o">-</span> <span class="n">n</span> <span class="o">*</span> <span class="n">np</span><span class="o">.</span><span class="n">log</span><span class="p">(</span><span class="n">d_min</span> <span class="o">-</span> <span class="mf">0.5</span><span class="p">))</span> <span class="o">+</span> <span class="mi">1</span></div>


<div class="viewcode-block" id="update_Sx"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.update_Sx">[docs]</a><span class="k">def</span> <span class="nf">update_Sx</span><span class="p">(</span><span class="n">S_old</span><span class="p">,</span> <span class="n">n_old</span><span class="p">,</span> <span class="n">d_old</span><span class="p">,</span> <span class="n">d_new</span><span class="p">,</span> <span class="n">d_min</span><span class="p">):</span>
    <span class="sd">&quot;&quot;&quot;</span>
<span class="sd">    Update on the sum of log degrees S_d and n based on degree distribution resulting from inserting or deleting</span>
<span class="sd">    a single edge.</span>
<span class="sd">    &quot;&quot;&quot;</span>

    <span class="n">old_in_range</span> <span class="o">=</span> <span class="n">d_old</span> <span class="o">&gt;=</span> <span class="n">d_min</span>
    <span class="n">new_in_range</span> <span class="o">=</span> <span class="n">d_new</span> <span class="o">&gt;=</span> <span class="n">d_min</span>

    <span class="n">d_old_in_range</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">multiply</span><span class="p">(</span><span class="n">d_old</span><span class="p">,</span> <span class="n">old_in_range</span><span class="p">)</span>
    <span class="n">d_new_in_range</span> <span class="o">=</span> <span class="n">np</span><span class="o">.</span><span class="n">multiply</span><span class="p">(</span><span class="n">d_new</span><span class="p">,</span> <span class="n">new_in_range</span><span class="p">)</span>

    <span class="n">new_S_d</span> <span class="o">=</span> <span class="n">S_old</span> <span class="o">-</span> <span class="n">np</span><span class="o">.</span><span class="n">log</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">maximum</span><span class="p">(</span><span class="n">d_old_in_range</span><span class="p">,</span> <span class="mi">1</span><span class="p">))</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="mi">1</span><span class="p">)</span> <span class="o">+</span> <span class="n">np</span><span class="o">.</span><span class="n">log</span><span class="p">(</span><span class="n">np</span><span class="o">.</span><span class="n">maximum</span><span class="p">(</span><span class="n">d_new_in_range</span><span class="p">,</span> <span class="mi">1</span><span class="p">))</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="mi">1</span><span class="p">)</span>
    <span class="n">new_n</span> <span class="o">=</span> <span class="n">n_old</span> <span class="o">-</span> <span class="n">np</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="n">old_in_range</span><span class="p">,</span> <span class="mi">1</span><span class="p">)</span> <span class="o">+</span> <span class="n">np</span><span class="o">.</span><span class="n">sum</span><span class="p">(</span><span class="n">new_in_range</span><span class="p">,</span> <span class="mi">1</span><span class="p">)</span>

    <span class="k">return</span> <span class="n">new_S_d</span><span class="p">,</span> <span class="n">new_n</span></div>


<div class="viewcode-block" id="compute_log_likelihood"><a class="viewcode-back" href="../../../../source/deeprobust.graph.targeted_attack.html#deeprobust.graph.targeted_attack.nettack.compute_log_likelihood">[docs]</a><span class="k">def</span> <span class="nf">compute_log_likelihood</span><span class="p">(</span><span class="n">n</span><span class="p">,</span> <span class="n">alpha</span><span class="p">,</span> <span class="n">S_d</span><span class="p">,</span> <span class="n">d_min</span><span class="p">):</span>
    <span class="sd">&quot;&quot;&quot;</span>
<span class="sd">    Compute log likelihood of the powerlaw fit.</span>

<span class="sd">    &quot;&quot;&quot;</span>

    <span class="k">return</span> <span class="n">n</span> <span class="o">*</span> <span class="n">np</span><span class="o">.</span><span class="n">log</span><span class="p">(</span><span class="n">alpha</span><span class="p">)</span> <span class="o">+</span> <span class="n">n</span> <span class="o">*</span> <span class="n">alpha</span> <span class="o">*</span> <span class="n">np</span><span class="o">.</span><span class="n">log</span><span class="p">(</span><span class="n">d_min</span><span class="p">)</span> <span class="o">+</span> <span class="p">(</span><span class="n">alpha</span> <span class="o">+</span> <span class="mi">1</span><span class="p">)</span> <span class="o">*</span> <span class="n">S_d</span></div>

<span class="k">def</span> <span class="nf">filter_chisquare</span><span class="p">(</span><span class="n">ll_ratios</span><span class="p">,</span> <span class="n">cutoff</span><span class="p">):</span>
    <span class="k">return</span> <span class="n">ll_ratios</span> <span class="o">&lt;</span> <span class="n">cutoff</span>
</pre></div>

           </div>
           
          </div>
          <footer>
  

  <hr/>

  <div role="contentinfo">
    <p>
        
        &copy; Copyright 

    </p>
  </div>
    
    
    
    Built with <a href="http://sphinx-doc.org/">Sphinx</a> using a
    
    <a href="https://github.com/rtfd/sphinx_rtd_theme">theme</a>
    
    provided by <a href="https://readthedocs.org">Read the Docs</a>. 

</footer>

        </div>
      </div>

    </section>

  </div>
  

  <script type="text/javascript">
      jQuery(function () {
          SphinxRtdTheme.Navigation.enable(true);
      });
  </script>

  
  
    
   

</body>
</html>