{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import math\n",
    "from tqdm import tqdm\n",
    "import torch\n",
    "from itertools import product\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[array([1, 1, 1]), array([ 1,  1, -1]), array([ 1, -1,  1]), array([ 1, -1, -1]), array([-1,  1,  1]), array([-1,  1, -1]), array([-1, -1,  1]), array([-1, -1, -1])]\n"
     ]
    }
   ],
   "source": [
    "\n",
    "\n",
    "def generate_pm_vectors(d):\n",
    "    return [np.array(list(vec)) for vec in product([1, -1], repeat=d)]\n",
    "\n",
    "# Example usage:\n",
    "d = 3\n",
    "vectors = generate_pm_vectors(d)\n",
    "print(vectors)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n",
      "[0 0 0]\n",
      "[0 0 2]\n",
      "[0 2 0]\n",
      "[0 2 2]\n",
      "[2 0 0]\n",
      "[2 0 2]\n",
      "[2 2 0]\n",
      "[2 2 2]\n",
      "1\n",
      "[ 0  0 -2]\n",
      "[0 0 0]\n",
      "[ 0  2 -2]\n",
      "[0 2 0]\n",
      "[ 2  0 -2]\n",
      "[2 0 0]\n",
      "[ 2  2 -2]\n",
      "[2 2 0]\n",
      "2\n",
      "[ 0 -2  0]\n",
      "[ 0 -2  2]\n",
      "[0 0 0]\n",
      "[0 0 2]\n",
      "[ 2 -2  0]\n",
      "[ 2 -2  2]\n",
      "[2 0 0]\n",
      "[2 0 2]\n",
      "3\n",
      "[ 0 -2 -2]\n",
      "[ 0 -2  0]\n",
      "[ 0  0 -2]\n",
      "[0 0 0]\n",
      "[ 2 -2 -2]\n",
      "[ 2 -2  0]\n",
      "[ 2  0 -2]\n",
      "[2 0 0]\n",
      "4\n",
      "[-2  0  0]\n",
      "[-2  0  2]\n",
      "[-2  2  0]\n",
      "[-2  2  2]\n",
      "[0 0 0]\n",
      "[0 0 2]\n",
      "[0 2 0]\n",
      "[0 2 2]\n",
      "5\n",
      "[-2  0 -2]\n",
      "[-2  0  0]\n",
      "[-2  2 -2]\n",
      "[-2  2  0]\n",
      "[ 0  0 -2]\n",
      "[0 0 0]\n",
      "[ 0  2 -2]\n",
      "[0 2 0]\n",
      "6\n",
      "[-2 -2  0]\n",
      "[-2 -2  2]\n",
      "[-2  0  0]\n",
      "[-2  0  2]\n",
      "[ 0 -2  0]\n",
      "[ 0 -2  2]\n",
      "[0 0 0]\n",
      "[0 0 2]\n",
      "7\n",
      "[-2 -2 -2]\n",
      "[-2 -2  0]\n",
      "[-2  0 -2]\n",
      "[-2  0  0]\n",
      "[ 0 -2 -2]\n",
      "[ 0 -2  0]\n",
      "[ 0  0 -2]\n",
      "[0 0 0]\n"
     ]
    }
   ],
   "source": [
    "for i in range(2**d):\n",
    "    print(i)\n",
    "    for j in range(2**d):\n",
    "        print(vectors[i] - vectors[j])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Ring():\n",
    "    def __init__(self, n_nodes):\n",
    "        self.w = torch.zeros((n_nodes, n_nodes))\n",
    "\n",
    "        for i in range(n_nodes):\n",
    "            self.w[i,i] = 1/3\n",
    "            self.w[i, (i+1)%n_nodes] = 1/3\n",
    "            self.w[i, (i-1)%n_nodes] = 1/3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Hypercube():\n",
    "    def __init__(self, n_nodes):\n",
    "        self.w = torch.zeros((n_nodes, n_nodes))\n",
    "        \n",
    "        dim = int(math.log2(n_nodes))\n",
    "        \n",
    "        node_index = self.generate_pm_vectors(dim)\n",
    "        \n",
    "        for i in range(n_nodes):\n",
    "            for j in range(n_nodes):\n",
    "                if sum((node_index[i] - node_index[j])**2) == 4:\n",
    "                    self.w[i, j] = 1/(1 + dim)\n",
    "                if i == j:\n",
    "                    self.w[i, i] = 1/ (1 + dim)\n",
    "                \n",
    "    def generate_pm_vectors(self, d):\n",
    "        return [np.array(list(vec)) for vec in product([1, -1], repeat=d)]\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.2000, 0.2000, 0.2000, 0.0000, 0.2000, 0.0000, 0.0000, 0.0000, 0.2000,\n",
       "         0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000],\n",
       "        [0.2000, 0.2000, 0.0000, 0.2000, 0.0000, 0.2000, 0.0000, 0.0000, 0.0000,\n",
       "         0.2000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000],\n",
       "        [0.2000, 0.0000, 0.2000, 0.2000, 0.0000, 0.0000, 0.2000, 0.0000, 0.0000,\n",
       "         0.0000, 0.2000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000],\n",
       "        [0.0000, 0.2000, 0.2000, 0.2000, 0.0000, 0.0000, 0.0000, 0.2000, 0.0000,\n",
       "         0.0000, 0.0000, 0.2000, 0.0000, 0.0000, 0.0000, 0.0000],\n",
       "        [0.2000, 0.0000, 0.0000, 0.0000, 0.2000, 0.2000, 0.2000, 0.0000, 0.0000,\n",
       "         0.0000, 0.0000, 0.0000, 0.2000, 0.0000, 0.0000, 0.0000],\n",
       "        [0.0000, 0.2000, 0.0000, 0.0000, 0.2000, 0.2000, 0.0000, 0.2000, 0.0000,\n",
       "         0.0000, 0.0000, 0.0000, 0.0000, 0.2000, 0.0000, 0.0000],\n",
       "        [0.0000, 0.0000, 0.2000, 0.0000, 0.2000, 0.0000, 0.2000, 0.2000, 0.0000,\n",
       "         0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.2000, 0.0000],\n",
       "        [0.0000, 0.0000, 0.0000, 0.2000, 0.0000, 0.2000, 0.2000, 0.2000, 0.0000,\n",
       "         0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.2000],\n",
       "        [0.2000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.2000,\n",
       "         0.2000, 0.2000, 0.0000, 0.2000, 0.0000, 0.0000, 0.0000],\n",
       "        [0.0000, 0.2000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.2000,\n",
       "         0.2000, 0.0000, 0.2000, 0.0000, 0.2000, 0.0000, 0.0000],\n",
       "        [0.0000, 0.0000, 0.2000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.2000,\n",
       "         0.0000, 0.2000, 0.2000, 0.0000, 0.0000, 0.2000, 0.0000],\n",
       "        [0.0000, 0.0000, 0.0000, 0.2000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
       "         0.2000, 0.2000, 0.2000, 0.0000, 0.0000, 0.0000, 0.2000],\n",
       "        [0.0000, 0.0000, 0.0000, 0.0000, 0.2000, 0.0000, 0.0000, 0.0000, 0.2000,\n",
       "         0.0000, 0.0000, 0.0000, 0.2000, 0.2000, 0.2000, 0.0000],\n",
       "        [0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.2000, 0.0000, 0.0000, 0.0000,\n",
       "         0.2000, 0.0000, 0.0000, 0.2000, 0.2000, 0.0000, 0.2000],\n",
       "        [0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.2000, 0.0000, 0.0000,\n",
       "         0.0000, 0.2000, 0.0000, 0.2000, 0.0000, 0.2000, 0.2000],\n",
       "        [0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.2000, 0.0000,\n",
       "         0.0000, 0.0000, 0.2000, 0.0000, 0.2000, 0.2000, 0.2000]])"
      ]
     },
     "execution_count": 48,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "graph = Hypercube(16)\n",
    "graph.w"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Star():\n",
    "    def __init__(self, n_nodes):\n",
    "        self.w = torch.zeros((n_nodes, n_nodes))\n",
    "        \n",
    "        self.w[0,0] = 1/n_nodes\n",
    "        for i in range(n_nodes):\n",
    "            if i == 0:\n",
    "                for j in range(n_nodes):\n",
    "                    self.w[i,j] = 1 / n_nodes\n",
    "            else:\n",
    "                self.w[i, 0] = 1 / n_nodes\n",
    "                self.w[0, i] = 1 / n_nodes\n",
    "                self.w[i, i] = 1 - 1 / n_nodes\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.2000, 0.2000, 0.2000, 0.2000, 0.2000],\n",
       "        [0.2000, 0.8000, 0.0000, 0.0000, 0.0000],\n",
       "        [0.2000, 0.0000, 0.8000, 0.0000, 0.0000],\n",
       "        [0.2000, 0.0000, 0.0000, 0.8000, 0.0000],\n",
       "        [0.2000, 0.0000, 0.0000, 0.0000, 0.8000]])"
      ]
     },
     "execution_count": 54,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "graph = Star(5)\n",
    "graph.w"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Ring2():\n",
    "    def __init__(self, n_nodes):\n",
    "        self.w = torch.zeros((n_nodes, n_nodes))\n",
    "\n",
    "        for i in range(n_nodes):\n",
    "            #self.w[i,i] = 1/3\n",
    "            self.w[i, (i+1)%n_nodes] = 1/2\n",
    "            self.w[i, (i-1)%n_nodes] = 1/2\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
