{
  "cells": [
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {},
      "source": [
        "# Graph Separation\n",
        "This code generates the data for Figure 1(a) and the plot of Figure 4."
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 1,
      "metadata": {
        "id": "QOMalDxn7tK8"
      },
      "outputs": [],
      "source": [
        "#number of graphs to check on (to reconstruct results in paper set graph_num to the maximal value of 600)\n",
        "graph_num=5"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 2,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "F1op-CbyLuN4",
        "outputId": "ecacb7d7-5c17-4746-d88b-494f07478fcc"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "2.0.1+cu118\n",
            "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m10.2/10.2 MB\u001b[0m \u001b[31m103.0 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m4.8/4.8 MB\u001b[0m \u001b[31m52.8 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[?25h  Installing build dependencies ... \u001b[?25l\u001b[?25hdone\n",
            "  Getting requirements to build wheel ... \u001b[?25l\u001b[?25hdone\n",
            "  Preparing metadata (pyproject.toml) ... \u001b[?25l\u001b[?25hdone\n",
            "  Building wheel for torch_geometric (pyproject.toml) ... \u001b[?25l\u001b[?25hdone\n"
          ]
        }
      ],
      "source": [
        "# Install required packages.\n",
        "import os\n",
        "import torch\n",
        "import torch.nn.functional as F\n",
        "\n",
        "torch.set_default_dtype(torch.float64)\n",
        "\n",
        "os.environ['TORCH'] = torch.__version__\n",
        "print(torch.__version__)\n",
        "\n",
        "%pip install -q torch-scatter -f https://data.pyg.org/whl/torch-${TORCH}.html\n",
        "%pip install -q torch-sparse -f https://data.pyg.org/whl/torch-${TORCH}.html\n",
        "%pip install -q git+https://github.com/pyg-team/pytorch_geometric.git\n",
        "\n"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "dszt2RUHE7lW"
      },
      "source": [
        "download TUdataset\n",
        "\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 3,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "imGrKO5YH11-",
        "outputId": "43d1a070-05b8-4f1f-bb7e-513dfae295b7"
      },
      "outputs": [
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "Downloading https://www.chrsmrrs.com/graphkerneldatasets/ENZYMES.zip\n",
            "Extracting /tmp/ENZYMES/ENZYMES/ENZYMES.zip\n",
            "Processing...\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "\n",
            "Dataset: ENZYMES(600):\n",
            "======================\n",
            "Number of graphs: 600\n",
            "Number of features: 3\n",
            "Number of classes: 6\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "Done!\n"
          ]
        }
      ],
      "source": [
        "\n",
        "from torch_geometric.datasets import TUDataset\n",
        "\n",
        "\n",
        "dataset = TUDataset(root='/tmp/ENZYMES', name='ENZYMES')\n",
        "print()\n",
        "print(f'Dataset: {dataset}:')\n",
        "print('======================')\n",
        "print(f'Number of graphs: {len(dataset)}')\n",
        "print(f'Number of features: {dataset.num_features}')\n",
        "print(f'Number of classes: {dataset.num_classes}')\n",
        "\n",
        "\n",
        "\n"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "_OWGw54wRd98"
      },
      "source": [
        "## A random Graph Neural Network (GNN)\n",
        "\n",
        "\n",
        "We use the **GCN layer** ([Kipf et al. (2017)](https://arxiv.org/abs/1609.02907)) which is defined as\n",
        "\n",
        "$$\n",
        "\\mathbf{x}_v^{(\\ell + 1)} = \\mathbf{W}^{(\\ell + 1)} \\sum_{w \\in \\mathcal{N}(v) \\, \\cup \\, \\{ v \\}} \\frac{1}{c_{w,v}} \\cdot \\mathbf{x}_w^{(\\ell)}\n",
        "$$\n",
        "\n",
        "where $\\mathbf{W}^{(\\ell + 1)}$ denotes a trainable weight matrix of shape `[num_output_features, num_input_features]` and $c_{w,v}$ refers to a fixed normalization coefficient for each edge.\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 4,
      "metadata": {
        "id": "fmXWs1dKIzD8"
      },
      "outputs": [],
      "source": [
        "from torch.nn import Linear\n",
        "from torch_geometric.nn import GCNConv\n",
        "\n",
        "class GCN(torch.nn.Module):\n",
        "    def __init__(self, hidden_channels,activation,doNormalize=False,num_features=1):\n",
        "        super().__init__()\n",
        "        torch.manual_seed(1234567)\n",
        "        self.doNormalize=doNormalize\n",
        "        self.activation=activation\n",
        "        self.w=torch.randn(1)\n",
        "        self.lin = Linear(num_features, hidden_channels)\n",
        "        self.conv1 = GCNConv(hidden_channels, hidden_channels)\n",
        "        self.conv2 = GCNConv(hidden_channels, hidden_channels)\n",
        "        self.conv3 = GCNConv(hidden_channels, hidden_channels)\n",
        "        \n",
        "    def Normalize(self,x):\n",
        "        N=torch.linalg.norm(x,ord=2)\n",
        "        n=torch.tensor(x.shape[0])\n",
        "        x=torch.sqrt(n)*x/N\n",
        "        \n",
        "        return x\n",
        "\n",
        "  \n",
        "\n",
        "    def forward(self, x, edge_index):\n",
        "        \n",
        "        x = self.lin(x)\n",
        "        if self.doNormalize:\n",
        "          x=self.Normalize(x)\n",
        "        x=self.activation(x)\n",
        "        x= self.conv1(x, edge_index)\n",
        "        if self.doNormalize:\n",
        "          x=self.Normalize(x)\n",
        "        \n",
        "        x=self.activation(x)\n",
        "        x = self.conv2(x, edge_index)\n",
        "        if self.doNormalize:\n",
        "          x=self.Normalize(x)\n",
        "        x=self.activation(x)\n",
        "      \n",
        "        \n",
        "   \n",
        "        x = self.conv3(x, edge_index)\n",
        "        x=self.Normalize(x)\n",
        "       \n",
        "        return x\n",
        "\n"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "gBRyJoJ5P6m2"
      },
      "source": [
        "WL convolutions"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 5,
      "metadata": {
        "id": "0nXNJZvtP_u9"
      },
      "outputs": [],
      "source": [
        "from torch.nn import Linear\n",
        "from torch_geometric.nn import WLConv\n",
        "\n",
        "\n",
        "class WL(torch.nn.Module):\n",
        "    def __init__(self):\n",
        "        super().__init__()\n",
        "        torch.manual_seed(1234567)\n",
        "        self.conv1 = WLConv()\n",
        "        self.conv2 = WLConv()\n",
        "        self.conv3 = WLConv()\n",
        "   \n",
        "    def forward(self, y, edge_index):\n",
        "        y=y.long()\n",
        "        y = self.conv1(y,edge_index)\n",
        "        y = self.conv2(y,edge_index)\n",
        "        y = self.conv3(y,edge_index)\n",
        "        return y\n",
        "\n"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "5tFUmjNsDRHZ"
      },
      "source": [
        "Function which, for given feature vectors, check number of equivalent pairs"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 6,
      "metadata": {
        "id": "ixnnW3U5DWkr"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "import scipy\n",
        "from scipy.spatial import distance_matrix\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "thresh=10**(-12)\n",
        "\n",
        "def separation_data(a,verbose=True,threshold=thresh):\n",
        "  a=a.detach().numpy()\n",
        "  D=distance_matrix(a, a)\n",
        "  if verbose:\n",
        "    plt.imshow(D<=threshold)\n",
        "    \n",
        "  #number of zeros in distance matrix\n",
        "  nz_all=np.sum(D<=threshold)\n",
        "  #nz number of zeros in lower triangular matrix\n",
        "  n=a.shape[0]\n",
        "  nz=(nz_all-n)/2\n",
        "  ds=np.sort(D,axis=None)\n",
        "  if nz_all<n**2:\n",
        "    next_smallest=ds[nz_all]\n",
        "  else:\n",
        "    next_smallest=-1\n",
        "  if verbose:\n",
        "    display('number of zeros')\n",
        "    display(nz)\n",
        "    display('next smalles value')\n",
        "    display(next_smallest)\n",
        "    display('largest value of distance matrix')\n",
        "    display(np.max(D))\n",
        "  out=[nz,next_smallest,ds]\n",
        "  return out\n",
        "\n",
        "\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 7,
      "metadata": {
        "id": "Tffk9tD-Y2S7"
      },
      "outputs": [],
      "source": [
        "def wl_on_dataset(graph_num,dataset):\n",
        "  tot_zero_wl=-torch.ones(graph_num)\n",
        "  next_smallest_wl=-torch.ones(graph_num)\n",
        "  for i in range(graph_num):\n",
        "    data=dataset[i]\n",
        "    #integer encoding of x using degree\n",
        "    deg = degree(data.edge_index[0], data.num_nodes)\n",
        "    encoding=deg\n",
        "    #WL\n",
        "    model = WL()\n",
        "    model.eval()\n",
        "    h = model(encoding, data.edge_index)\n",
        "    output=separation_data(torch.unsqueeze(h, 1),verbose=False)\n",
        "    tot_zero_wl[i]=output[0]\n",
        "    next_smallest_wl[i]=output[1]\n",
        "  out=[tot_zero_wl,next_smallest_wl]\n",
        "  return out"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 8,
      "metadata": {
        "id": "q_4np7DsZziz"
      },
      "outputs": [],
      "source": [
        "def gnn_on_dataset(graph_num,dataset,hidden,activation,doNormalize=False):\n",
        "  N=len(hidden)\n",
        "  tot_zero=-torch.ones(N,graph_num)\n",
        "  next_smallest=-torch.ones(N,graph_num)\n",
        "  ds_all=[]\n",
        "  for j in range(N):\n",
        "    ds_all_loc=[]\n",
        "    for i in range(graph_num):\n",
        "      data=dataset[i]\n",
        "      #integer encoding of x using degree\n",
        "      deg = degree(data.edge_index[0], data.num_nodes)\n",
        "      data=Data(x=torch.unsqueeze(deg, 1),edge_index=data.edge_index)\n",
        "      model = GCN(hidden_channels=hidden[j],activation=activation,doNormalize=doNormalize,num_features=1)\n",
        "      model.eval()\n",
        "      h = model(data.x.double(), data.edge_index)\n",
        "      output=separation_data(h,verbose=False)\n",
        "      tot_zero[j,i]=output[0]\n",
        "      next_smallest[j,i]=output[1]\n",
        "      ds_all_loc=np.concatenate((ds_all_loc,output[2]),-1)\n",
        "      #ds=output[2]\n",
        "      #ds_all=torch.cat((ds_all,ds),0)\n",
        "    ds_all.append(ds_all_loc)\n",
        "  out=[tot_zero,next_smallest,ds_all]\n",
        "  return out\n",
        "\n",
        "\n"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "XhO8QDgYf_Q8"
      },
      "source": [
        "Plot number of zeros as function of hidden channels, SiLU and leaky_relu activation"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "mHeOtI0xfICy"
      },
      "source": [
        "Function to check agreement of wl classification with gnn classification"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 9,
      "metadata": {
        "id": "E6RZlrLsan47"
      },
      "outputs": [],
      "source": [
        "\n",
        "def check_agreement(tot_zero_wl,tot_zero_GNN):\n",
        "  I=tot_zero_GNN.shape[0]\n",
        "  J=tot_zero_GNN.shape[1]\n",
        "  diff_GNN=torch.zeros(I,J)\n",
        "  total_err=torch.zeros(I)\n",
        "  total_err_minus=torch.zeros(I)\n",
        "  for i in range(I):\n",
        "    diff_GNN[i,:]=tot_zero_GNN[i,:]-tot_zero_wl\n",
        "    total_err[i]=torch.sum(diff_GNN[i,:]>0.5)\n",
        "    total_err_minus[i]=torch.sum(diff_GNN[i,:]<-0.5)\n",
        "  display('total number of positive errors',total_err)\n",
        "  display('total number of negative errors',total_err_minus)\n",
        "  return total_err\n",
        "\n",
        "\n"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "4cV-HZ--3cnc"
      },
      "source": [
        "function to extract statistics re smallest non-zero distance over all graphs"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 10,
      "metadata": {
        "id": "iC0ICFq03pdM"
      },
      "outputs": [],
      "source": [
        "def next_smallest_statistics(next_smallest):\n",
        "  abs_next=torch.abs(next_smallest)\n",
        "  m=torch.min(abs_next,axis=1)\n",
        "  med=torch.median(abs_next,axis=1)\n",
        "  display(m.values)\n",
        "  display(med.values)\n",
        "  return [m.values, med.values]"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "S24A_X-TETxW"
      },
      "source": [
        "Run graph separation experiment with different activations"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 11,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 767
        },
        "id": "FfdPacI9Voey",
        "outputId": "59f6b88e-8dc4-4641-e2e0-41d29e5c51e4"
      },
      "outputs": [
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'leaky relu'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0001, 0.0008, 0.0004, 0.0005])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0011, 0.0048, 0.0005, 0.0008])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of positive errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of negative errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'Hardtanh'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of positive errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of negative errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0001, 0.0042, 0.0026, 0.0028])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0011, 0.0078, 0.0040, 0.0042])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'SiLU'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of positive errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of negative errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0003, 0.0014, 0.0016, 0.0017])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0006, 0.0054, 0.0027, 0.0028])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'Tanh'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of positive errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of negative errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([5.7707e-06, 1.8284e-03, 1.2737e-03, 1.4427e-03])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0016, 0.0057, 0.0030, 0.0029])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'sin'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of positive errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of negative errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0003, 0.0022, 0.0041, 0.0051])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0015, 0.0072, 0.0097, 0.0111])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'relu'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of positive errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'total number of negative errors'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0., 0., 0., 0.])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0001, 0.0008, 0.0004, 0.0004])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0011, 0.0048, 0.0005, 0.0007])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "from torch_geometric.utils import degree\n",
        "from torch_geometric.data import Data\n",
        "\n",
        "\n",
        "hidden=[1,10,50,100]\n",
        "#run wl\n",
        "out= wl_on_dataset(graph_num,dataset)\n",
        "tot_zero_wl=out[0]\n",
        "next_smallest_wl=out[1]\n",
        "\n",
        "#run various gnns\n",
        "display('leaky relu')\n",
        "activation=torch.nn.LeakyReLU()\n",
        "out=gnn_on_dataset(graph_num,dataset,hidden,activation)\n",
        "#tot_zero_leaky=out[0]\n",
        "#next_smallest_leaky=out[1]\n",
        "next_smallest_statistics(out[1])\n",
        "check_agreement(tot_zero_wl,out[0])\n",
        "\n",
        "\n",
        "\n",
        "display('Hardtanh')\n",
        "activation=torch.nn.Hardtanh()\n",
        "out=gnn_on_dataset(graph_num,dataset,hidden,activation)\n",
        "total_err_hardtanh=check_agreement(tot_zero_wl,out[0])\n",
        "next_smallest_statistics(out[1])\n",
        "outHardtanh=out\n",
        "\n",
        "display('SiLU')\n",
        "activation=torch.nn.SiLU()\n",
        "out=gnn_on_dataset(graph_num,dataset,hidden,activation)\n",
        "total_err_silu=check_agreement(tot_zero_wl,out[0])\n",
        "next_smallest_statistics(out[1])\n",
        "outSilu=out\n",
        "\n",
        "display('Tanh')\n",
        "activation=torch.nn.Tanh()\n",
        "out=gnn_on_dataset(graph_num,dataset,hidden,activation)\n",
        "total_err_tanh=check_agreement(tot_zero_wl,out[0])\n",
        "next_smallest_statistics(out[1])\n",
        "outTanh=out\n",
        "\n",
        "display('sin')\n",
        "activation=torch.sin\n",
        "out=gnn_on_dataset(graph_num,dataset,hidden,activation)\n",
        "total_err_sin=check_agreement(tot_zero_wl,out[0])\n",
        "next_smallest_statistics(out[1])\n",
        "outSin=out\n",
        "\n",
        "display('relu')\n",
        "activation=torch.nn.ReLU()\n",
        "out=gnn_on_dataset(graph_num,dataset,hidden,activation)\n",
        "check_agreement(tot_zero_wl,out[0])\n",
        "next_smallest_statistics(out[1])\n",
        "outRelu=out\n",
        "\n"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "gPuvrgobASEF"
      },
      "source": [
        "For SiLU activation: plot the smallest and median non-zero distance as a function of feature dimension\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 12,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 176
        },
        "id": "aN03qWo0xu2a",
        "outputId": "ae80bd1f-9bc1-4ab7-d8c3-b869bd341855"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "array([  1,   5,  10,  15,  20,  25,  30,  35,  40,  45,  50,  55,  60,\n",
              "        65,  70,  75,  80,  85,  90,  95, 100])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'SiLU'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0003, 0.0007, 0.0014, 0.0025, 0.0011, 0.0020, 0.0021, 0.0012, 0.0023,\n",
              "        0.0025, 0.0016, 0.0027, 0.0022, 0.0020, 0.0025, 0.0020, 0.0020, 0.0018,\n",
              "        0.0013, 0.0016, 0.0017])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "tensor([0.0006, 0.0013, 0.0054, 0.0044, 0.0023, 0.0035, 0.0025, 0.0022, 0.0034,\n",
              "        0.0028, 0.0027, 0.0037, 0.0037, 0.0026, 0.0035, 0.0030, 0.0028, 0.0029,\n",
              "        0.0024, 0.0030, 0.0028])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "#hidden=torch.arange(0,101,5)\n",
        "#hidden[0]=1\n",
        "#hidden=hidden.long()\n",
        "hidden=np.arange(0, 101, 5)\n",
        "hidden[0]=1\n",
        "display(hidden)\n",
        "\n",
        "display('SiLU')\n",
        "activation=torch.nn.SiLU()\n",
        "out=gnn_on_dataset(graph_num,dataset,hidden,activation)\n",
        "output=next_smallest_statistics(out[1])\n",
        "\n",
        "\n",
        "\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 13,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 354
        },
        "id": "kE1NrEHvH02b",
        "outputId": "eaaac25d-7638-4ad3-ad65-e6d7eadde7f5"
      },
      "outputs": [
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'min, second method'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "array([0.00025797, 0.00066984, 0.0014328 , 0.00245574, 0.00110079,\n",
              "       0.00200638, 0.00207946, 0.00120236, 0.00230992, 0.00251066,\n",
              "       0.00164917, 0.00274883, 0.00224376, 0.00203441, 0.00250436,\n",
              "       0.00195915, 0.00196415, 0.00182096, 0.00125217, 0.00158757,\n",
              "       0.00171513])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'median:'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "array([0.20514865, 0.08958551, 0.15088896, 0.15274666, 0.12690025,\n",
              "       0.13191599, 0.13715252, 0.13657013, 0.16320739, 0.14189913,\n",
              "       0.13125087, 0.14131119, 0.1284673 , 0.13018713, 0.14366709,\n",
              "       0.13125922, 0.13174146, 0.12864465, 0.13686929, 0.13349034,\n",
              "       0.13009086])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            },
            "text/plain": [
              "'max:'"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "array([0.95355519, 0.3182623 , 0.60483922, 0.62598913, 0.52808515,\n",
              "       0.54947882, 0.57610366, 0.57166071, 0.66990632, 0.58400215,\n",
              "       0.54676983, 0.56708307, 0.51685221, 0.54521355, 0.56705007,\n",
              "       0.52837253, 0.54334855, 0.52071133, 0.57160062, 0.5566538 ,\n",
              "       0.53429354])"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "text/plain": [
              "(3628,)"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "ds_list=out[2]\n",
        "nh=len(hidden)\n",
        "min_dist_nz=-np.ones(nh)\n",
        "median_nz=-np.ones(nh)\n",
        "max_nz=-np.ones(nh)\n",
        "for i in range(nh):\n",
        "  ds=ds_list[i]\n",
        "  ds=np.sort(ds)\n",
        "  #remove thresholded values\n",
        "  num_z=np.sum(ds<=thresh)\n",
        "  ds_nz=ds[num_z:]\n",
        "  min_dist_nz[i]=np.amin(ds_nz)\n",
        "  median_nz[i] =np.median(ds_nz)\n",
        "  max_nz[i] =np.max(ds_nz)\n",
        "\n",
        "display('min, second method')\n",
        "display(min_dist_nz)\n",
        "display('median:',median_nz)\n",
        "display('max:',max_nz)\n",
        "display(ds.shape)\n",
        "        \n",
        "\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 14,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 436
        },
        "id": "tgY4AKXXFCO1",
        "outputId": "4669fefb-e841-4d05-c556-302633098734"
      },
      "outputs": [
        {
          "data": {
            "application/javascript": "\n    async function download(id, filename, size) {\n      if (!google.colab.kernel.accessAllowed) {\n        return;\n      }\n      const div = document.createElement('div');\n      const label = document.createElement('label');\n      label.textContent = `Downloading \"${filename}\": `;\n      div.appendChild(label);\n      const progress = document.createElement('progress');\n      progress.max = size;\n      div.appendChild(progress);\n      document.body.appendChild(div);\n\n      const buffers = [];\n      let downloaded = 0;\n\n      const channel = await google.colab.kernel.comms.open(id);\n      // Send a message to notify the kernel that we're ready.\n      channel.send({})\n\n      for await (const message of channel.messages) {\n        // Send a message to notify the kernel that we're ready.\n        channel.send({})\n        if (message.buffers) {\n          for (const buffer of message.buffers) {\n            buffers.push(buffer);\n            downloaded += buffer.byteLength;\n            progress.value = downloaded;\n          }\n        }\n      }\n      const blob = new Blob(buffers, {type: 'application/binary'});\n      const a = document.createElement('a');\n      a.href = window.URL.createObjectURL(blob);\n      a.download = filename;\n      div.appendChild(a);\n      a.click();\n      div.remove();\n    }\n  ",
            "text/plain": [
              "<IPython.core.display.Javascript object>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "application/javascript": "download(\"download_7ee0d7e7-cc40-4357-a760-db673a257408\", \"wl_numerics.pdf\", 13784)",
            "text/plain": [
              "<IPython.core.display.Javascript object>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "data": {
            "image/png": "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pMVro5rzyy+/YPTo0UhNTTVbpqTv57y8PIP0v/76CwDQoEEDq9qme72/++47/eixOaZeb1vAYImILLKzMz8AbSnvSYV/PZubDFxYeV5SaObMmYiOjkaDBg2wcOFCtG7dGt7e3vrDci+++CKOHz9u9qyw4uy3tcpjm2X1fGfPnsWaNWuM0kNDQ42CpW3btmHSpElQKpXYvn27wSFaHd17oU6dOggJCbH43Ka+kB0dHa1uuy1q2LAhrl69ipiYGOzfvx/Hjh3D4cOHsX//fnz44Yf47rvvMGLECKu3d+PGDQwePBg5OTl47733MHz4cAQGBkKpVMLOzg4xMTHo3r17hb6fC9O93s2aNTM5glVY27Zty7UtJcVgiagQuVSCZ+XotHZfbIe3tzccHR2Rk5ODzz77zOq1j3SHaRITE82WsZRniu5U6p9++snkIbxr164Va3sVzcvLCwqFAnl5efjzzz9N7oPu17y5w1zF8cEHH1i1evSpU6cwePBgSCQSREVFmV04WDeyWL9+faNL1JRGefRLebz/AO0h7F69eukPc2dkZGDx4sWYN28e3njjDfTv39/iBeUL++WXX5CTk4P+/fvj008/Ncov6/ez7uzCK1euGMxNMkf3eoeEhGDZsmVl2paKwmCJqJCyONWeTJNKpejatSu2bt2KDRs2GCwrYknHjh0hkUhw5swZXLlyxWik4dy5c8U6BAdAvzaTqVGP3bt3IyUlpVjbq2gymQzt27fHvn37sHr1apOnhOuWX3jppZcqpE1//vknevfujezsbHz77bcm5/fodOnSBfb29jh48CDu3r1rNOG3pMqjX1q2bAmlUomUlBTExMQYzU+6c+cOYmJiSt12V1dXfPDBB4iMjERaWhp+//13/URv3YhnQUGBybqW3s9CCJMLPJdGjx49sHv3bnz//feYMGFCkfOWevbsiffffx9bt27FZ599VuwJ8bbg2fgJTURPhblz58Le3h7vvvsu1qxZY3Ji64ULF7B582b944CAAPTv3x8ajQYTJkwwmHfx4MEDTJw4sdiLKOomri5dutQg/erVqxg/fnyxtlVZ3nnnHQDAihUrjM46XL16NbZu3Qq5XI633nqr3NuSmpqKnj174u7du/jXv/6ln0RtTrVq1fDmm28iKysLffr0wfnz543K5OXlYevWrbhy5Uqx2lLW/eLo6Ihx48YB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N4+IiIieQzYVLAHArFmzAAALFy7EmTNn9OmpqamYOHEiAGDSpEnFXoSSiIiIqCQkQghRHhs+c+aMPrgBgOvXryMlJQV+fn4Glx2Jjo5GjRo1DOq+9dZbWLJkCeRyObp06QJnZ2fs27cPaWlpCAkJwZ49e0p9Id2ylJGRATc3N6Snp3PiORER0VPC2u/vcls6ICMjA7GxsUbpycnJSE5O1j/Oy8szKhMZGYmQkBAsX74cx44dg0qlQu3atTFjxgy8/fbbsLe3L69mExERERkot5Gl5wlHloiIiJ4+1n5/29ycJSIiIiJbwmCJiIiIyAIGS0REREQWMFgiIiIisoDBEhEREZEFDJaIiIiILGCwRERERGQBgyUiIiIiCxgsEREREVnAYImIiIjIAgZLRERERBYwWCIiIiKygMESERERkQUMloiIiIgsYLBEREREZAGDJSIiIiILGCwRERERWcBgiYiIiMgCBktEREREFjBYIiIiIrKAwRIRERGRBQyWiIiIiCxgsERERERkAYMlIiIiIgsYLBERERFZwGCJiIiIyAIGS0REREQWMFgiIiIisoDBEhEREZEFDJaIiIiILGCwRERERGQBgyUiIiIiCxgsEREREVnAYImIiIjIAgZLRERERBY8k8HSH3/8gfHjx6NFixaQy+UIDAwsVTkiIiJ6fskquwHl4eLFi9i2bRvatGkDIQQePHhQqnJERET0/HomR5b69OmD5ORkbN68GW3bti11OSIiInp+PZPBkp2ddbtlbTkiIiJ6fpUoWrh69SqWLl2KiIgIBAcHQyaTQSKRYMGCBVbVj4qKQmhoKDw8PODs7IymTZti0aJFUKlUJWkOERERUbkp0ZylFStWIDIyskRPOGXKFERGRkImk6Fz585QKpXYv38/pk+fjl9++QUxMTFwdHQs0baJiIiIylqJRpaaNGmCadOmYd26dbh8+TJGjhxpVb0tW7YgMjISSqUSsbGx2L17NzZt2oRr164hODgYR44cwZw5cwzqrF69GhKJpMjbxo0bS7IrRERERBaVaGRp7NixBo+tnfvz8ccfAwBmzJiBFi1a6NO9vb3x5ZdfokOHDli2bBnmzJkDNzc3AED//v3Rrl27Irft6+trbfOJiIiIrFZhSwfcuHEDp06dAgAMGzbMKL99+/bw9/dHUlISduzYgaFDhwIA3Nzc9IETERERUUWrsNPB4uPjAQCenp4ICgoyWaZVq1YGZW1VXl4eMjIyDG5ERET0bKqwkaWEhAQAQEBAgNky/v7+BmVLKjs7Gzt27AAA/Pnnn8jOztbPaWrdujVq1qxZrHJP+uSTTzBv3rxStZGIiIieDhUWLGVmZgIAnJ2dzZZRKpUAUOqRmrt372LQoEEGabrHq1atQkRERLHKPWnmzJmYOnWq/nFGRoY+0CMiIqJnyzN5uZPAwEAIIcqs3JMUCgUUCkVJmkZERERPmQqbs+Ti4gIAyMrKMlvm4cOHAABXV9cKaRMRERFRUSosWAoMDAQAJCUlmS2jy9OVJSIiIqpsFRYsNW/eHACQmppqdgL36dOnAcBgDSYiIiKiylRhwZKfnx9at24NAFi/fr1R/pEjR5CUlASFQoFevXpVVLOIiIiILKqwYAkAZs2aBQBYuHAhzpw5o09PTU3FxIkTAQCTJk3iIpRERERkMySiBKeDnTlzRh/cAMD169eRkpICPz8/g8uOREdHo0aNGgZ133rrLSxZsgRyuRxdunSBs7Mz9u3bh7S0NISEhGDPnj1P3YV0MzIy4ObmhvT0dE5OJyIiekpY+/1doqUDMjIyEBsba5SenJyM5ORk/eO8vDyjMpGRkQgJCcHy5ctx7NgxqFQq1K5dGzNmzMDbb78Ne3v7kjSJiIiIqFyUaGSJDHFkiYiI6Olj7fd3hc5ZIiIiInraMFgiIiIisoDBEhEREZEFDJaIiIiILGCwRERERGQBgyUiIiIiCxgsEREREVnAYImIiIjIAgZLRERERBYwWCIiIiKygMESERERkQUMloiIiIgsYLBEREREZAGDJSIiIiILZJXdALJs/vz5WLBgQbk/z4kTJ9C8eXOryoaFhWH79u3l3CIgLy/P6rK1atXCjRs3yrE1QLNmzRAbG2tV2Xv37sHPz69c2wMAo0ePxsqVK60qGxMTgz59+pRzi4AvvvgCEyZMsKrsokWLMGfOnHJuEXDo0CG0a9fOqrJDhw7F5s2by7lFQEZGBhQKhVVlGzVqhOvXr5drexo2bIizZ89aVTYjIwNVqlQp1/YA2tdi9erVVpU9dOgQunXrVr4NgvY9+9Zbb1lVNjIyEu+99145t0j7v92pUyerykZEROB///tfObdI+xno6upqVdlmzZrh8uXLFsvMnj27Qj4rzGGwZOPUajXy8/PL/XmEEFaXLSgoqJA2FUd+fn65t0mlUhWrfEX0UUFBgdVlhRAV0iaNRmN1Wb6/rcP3d9Eq6v2tVquLVZbv76KpVKoi21Scfi8PPAxHREREZAGDJSIiIiILGCwRERERWcBgiYiIiMgCTvC2cVKpFPb29uX+PBKJxOqyMpmsQtpUHPb29uXeJrlcXqzyFdFHMpn1/8ISiaRC2mRnZ/1vML6/rcP3d9Eq6v0tlUqLVZbv76LJ5fIi21Scfi8PElGcafRkUkZGBtzc3JCenm71qZJERERUuaz9/uZhOCIiIiILGCwRERERWcBgiYiIiMgCBktEREREFjBYIiIiIrKAwRIRERGRBQyWiIiIiCxgsERERERkAYMlIiIiIgsYLBERERFZwGCJiIiIyAJeSLcM6C6vl5GRUcktISIiImvpvreLukwug6UykJmZCQDw9/ev5JYQERFRcWVmZsLNzc1svkQUFU5RkTQaDW7evAkXFxdIJJISbycjIwP+/v5ISkqyePVjKj32dcVhX1cc9nXFYV9XnPLsayEEMjMz4ePjAzs78zOTOLJUBuzs7ODn51dm23N1deU/XwVhX1cc9nXFYV9XHPZ1xSmvvrY0oqTDCd5EREREFjBYIiIiIrKAwZINUSgUmDt3LhQKRWU35ZnHvq447OuKw76uOOzrimMLfc0J3kREREQWcGSJiIiIyAIGS0REREQWMFgiIiIisoDBkg2IiopCaGgoPDw84OzsjKZNm2LRokVQqVSV3bSnhkqlwr59+/Duu++idevWcHd3h1wuR/Xq1dG3b19s377dYv29e/eiV69e8Pb2hqOjIxo0aID3338fDx8+rKA9ePq99957kEgkkEgkWLBggdly7OuSyc/Px5IlS9C+fXt4enrCwcEBfn5+6NmzJ3766SeTddjXxff3339j0qRJqF+/PhwdHeHg4ICgoCCEh4fj3LlzZuuxr41dvXoVS5cuRUREBIKDgyGTyYr8fNApaX/+8ccfiIiIgJ+fHxQKBfz8/BAREYE///yzdDsjqFK99dZbAoCQyWSiW7duIiwsTLi7uwsAon379iI7O7uym/hU2LNnjwAgAIjq1auLV155Rbz22muiSZMm+vRx48YJjUZjVHfx4sUCgJBIJKJjx45i0KBBonr16gKAqF+/vrh3714l7NHT5ejRo8LOzk5IJBIBQMyfP99kOfZ1ySQlJYlGjRoJAMLb21v07t1bDB48WLz44ovCyclJDBgwwKgO+7r4Tpw4IVxcXAQA4evrK/r27Sv69+8vgoKC9J/TGzZsMKrHvjZN9/325M3c54NOSfvzyJEjwsnJSQAQjRs3FoMHDxaNGzcWAISzs7M4fvx4ifeFwVIlio6OFgCEUqkUcXFx+vR79+6J4OBgAUC88847ldjCp8e+ffvEgAEDxK+//mqU9+OPPwqpVCoAiDVr1hjknTlzRkgkEiGVSsWOHTv06VlZWaJLly4CgMkvInosKytL1K1bV/j6+op+/fqZ/TBkX5dMdna2aNCggQAgPvjgA5Gfn2+Qn5WVJeLj4w3S2Ncl88ILL+h/WBXuZ7VaLWbPni0ACHd3d5GTk6PPY1+b980334hp06aJdevWicuXL4uRI0cWGSyVtD+zsrKEj4+PACBmzpxpkDdz5kwBQPj7+5d4AILBUiVq3bq1ACAWLFhglHf48GEBQCgUCpGWllYJrXu2jBkzRgAQXbp0MUgfNGiQACDGjh1rVCcxMVHY2dkJAOLy5csV1dSnzuTJkwUAsX37dhEeHm72w5B9XTJz5szRf4Fbi31dfCkpKfqRj7t37xrlFxQUCEdHRwFAnDlzRp/Ovraepc8HnZL25/LlywUAUa9ePaFWqw3y1Gq1qFevngAgVq5cWaK2c85SJblx4wZOnToFABg2bJhRfvv27eHv74+8vDzs2LGjopv3zGnevDkAICkpSZ+Wn5+vn8tk6jWoWbMmQkJCAADR0dEV0Mqnz8GDB7F06VKMGjUKvXr1MluOfV0yKpUKK1asAAC8++67VtVhX5dMcRY89Pb2BsC+Lmul6U/d4yFDhhhdENfOzg6DBw8GAGzevLlEbWOwVEni4+MBAJ6enggKCjJZplWrVgZlqeSuXbsGAKhRo4Y+7ffff0d2djaAx339JL4G5j18+BCvv/46qlWrhi+++MJiWfZ1yZw5cwYpKSnw8fFBnTp1cP78ecybNw9vvPEGZsyYge3bt0Oj0RjUYV+XjFKpRIcOHQAAs2fPNjjBRqPR4IMPPkBOTg569uwJf39/AOzrslaa/tQ9Lq/XQVaiWlRqCQkJAICAgACzZXT/kLqyVDK3b9/G6tWrAQADBgzQp+v61d3dHS4uLibr8jUwb9q0aUhISEB0dDQ8PDwslmVfl8xvv/0GAPDz88OMGTOwaNEiiEIXXfj000/RvHlzbNmyRf9Zwr4uuW+++Qa9evXC119/je3bt6NVq1aQSqWIj4/HjRs3MHLkSCxbtkxfnn1dtkran5mZmUhNTQVg/jtVV+/evXvIysqCs7NzsdrGkaVKkpmZCQAWXzClUgkAyMjIqJA2PYsKCgowYsQIpKenIzg4GG+88YY+j69BycXExOCrr77CkCFD0K9fvyLLs69LRvcFEB8fj08//RQTJ07E1atXkZ6ejj179qBevXqIj4/HK6+8oh8JYV+XXP369XH8+HF069YNN27cwM8//4zNmzcjISEBderUQWhoKFxdXfXl2ddlq6T9qatnqa6u3pN1rcVgiZ5p48ePx759++Dl5YWNGzfC3t6+spv01EtPT8eYMWNQpUoVLF26tLKb80zTjSKpVCoMHToUy5YtQ7169eDq6oqXX34Ze/bsgYODAy5cuIAff/yxklv79Dt69CiCg4Nx4cIFrF+/Hrdv38b9+/fxyy+/QKVSYcyYMRgzZkxlN5MqAYOlSqIbYszKyjJbRrf4VuFfMmS9t956C9999x08PDz0v8IL42tQMlOmTEFycjKWLVumn+haFPZ1yRQ+FFF4VFQnICAAr7zyCgDtIn6F67CviyctLQ39+/fHvXv3sHnzZgwdOhTVqlWDh4cHevfujV27dsHJyQnff/89Dhw4AIB9XdZK2p+F/0/M1S28mGVJXgsGS5UkMDAQgOHZWU/S5enKkvXeeecdLFmyBO7u7oiJidGfDVeYrl/T0tIMhnEL42tgLDo6GjKZDF9++SVCQ0MNbrt27QIAfPfddwgNDcWQIUMAsK9LqlatWibvmypz69YtAOzrktq+fTvu3buHWrVqoW3btkb5hdN1gSn7umyVtD9dXFzg6ekJQLsCu6V63t7exZ6vBDBYqjS6L+/U1FSzE/9Onz4NAGjRokWFtetZ8N5772Hx4sVwc3NDTEyM2bMj6tevDycnJwCP+/pJfA1MKygowKFDh4xud+7cAQAkJibi0KFDOHHiBAD2dUm1aNECEokEAJCSkmKyjC5dNyeDfV0yui9ZS6MObm5uAID79+8DYF+XtdL0p+5xeb0ODJYqiZ+fH1q3bg0AWL9+vVH+kSNHkJSUBIVCYXH9GjI0Y8YM/Pvf/4abmxv27Nmj72NT7O3t9YcwTL0Gf/31F44dOwYA6N+/f/k0+CmUlpYGoV3Q1ugWHh4OAJg/fz6EEEhMTATAvi6p6tWro3379gAej2YUplKpcOjQIQBAmzZtALCvS8rX1xcAcOXKFaSnpxvlq1QqnDlzBgD0y72wr8tWafpT9/jHH380Wk5Do9Hor58YFhZWssaVaClLKhPmLneSkpLCy52UwPvvv6+/HMHJkyetqhMXF6dfWn/nzp36dF6qoGQsrdDLvi6ZvXv3CgDCw8PD4NpWKpVKvPnmmwKAcHFxEbdv39bnsa+L7+7du8LZ2VkAEIMGDRKZmZn6vLy8PPHPf/5TABByuVxcv35dn8e+tp41K3iXtD8LX+5k1qxZBnmzZs0SAISfnx8vd/K00l0qQi6Xix49eogBAwboL6QbEhLCC+la6eeff9ZfqqBVq1YiPDzc5M1U8Fn4oo2hoaHitddeEzVq1HjuL4JZEkV9GLKvS2b+/Pn6C7m++OKLIiwsTAQGBgoAwtHRUWzbts2oDvu6+P773/8KmUwmAIgqVaqIXr16iVdffVX4+voKAMLOzk6sWLHCqB772rS4uDjRtm1b/c3b21sftBROv3nzpkG9kvZn4QvpNmnSRAwZMkR/MXVeSPcZ8NNPP4mOHTsKV1dX4ejoKJo0aSIWLlwo8vLyKrtpT41Vq1aZvLr1k7eaNWuarL9nzx7Ro0cP4enpKRQKhahbt66YOXOmyMjIqNgdecpZ88uRfV0yu3fvFj179hSenp5CLpcLf39/ERERYfGaY+zr4jt79qyIiIgQtWrVEgqFQtjb24uaNWuK4cOHi9jYWLP12NfGDhw4YNXnckJCglHdkvbntWvXxKhRo4SPj4+Qy+XCx8dHjBo1Svzxxx+l2heJEIWWgyUiIiIiA5zgTURERGQBgyUiIiIiCxgsEREREVnAYImIiIjIAgZLRERERBYwWCIiIiKygMESERERkQUMloiIiIgsYLBEREREZAGDJSIiIiILGCwRERERWcBgiYiIiMgCBktEREREFvw/PKAcLnZMQcYAAAAASUVORK5CYII=",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "\n",
        "fontsize=\"16\"\n",
        "linewidth=\"5\"\n",
        "\n",
        "plt.rc('font', size=fontsize)          # controls default text sizes\n",
        "plt.rc('axes', titlesize=fontsize)     # fontsize of the axes title\n",
        "plt.rc('axes', labelsize=fontsize)    # fontsize of the x and y labels\n",
        "plt.rc('xtick', labelsize=fontsize)    # fontsize of the tick labels\n",
        "plt.rc('ytick', labelsize=fontsize)    # fontsize of the tick labels\n",
        "plt.rc('legend', fontsize=fontsize)    # legend fontsize\n",
        "plt.rc('figure', titlesize=fontsize)  # fontsize of the figure title\n",
        "\n",
        "\n",
        "fig=plt.figure()\n",
        "ax = fig.add_subplot()\n",
        "\n",
        "\n",
        "ax.set_yscale('log')\n",
        "plt.plot(hidden,np.ones(len(hidden)),'k-.',linewidth=linewidth)\n",
        "#plt.plot(hidden,output[1])     #median\n",
        "#plt.plot(hidden,output[0])     #min\n",
        "plt.plot(hidden,median_nz,linewidth=linewidth)     #median\n",
        "plt.plot(hidden,min_dist_nz,linewidth=linewidth)     #min\n",
        "plt.plot(hidden,thresh*np.ones(len(hidden)),'k--',linewidth=linewidth)\n",
        "ax.legend(['average feature norm', 'median non-zero distance','minimal non-zero distance','threshold'],fontsize=fontsize)\n",
        "doDownload=False\n",
        "if doDownload:\n",
        "  from google.colab import files \n",
        "  plt.savefig('wl_numerics.pdf')\n",
        "  files.download('wl_numerics.pdf') \n"
      ]
    }
  ],
  "metadata": {
    "accelerator": "GPU",
    "colab": {
      "gpuType": "T4",
      "provenance": []
    },
    "gpuClass": "standard",
    "kernelspec": {
      "display_name": "Python 3",
      "name": "python3"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}
