{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Populating the interactive namespace from numpy and matplotlib\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "Bad key \"text.kerning_factor\" on line 4 in\n",
      "/Users/csababoth/anaconda3/envs/tf1/lib/python3.7/site-packages/matplotlib/mpl-data/stylelib/_classic_test_patch.mplstyle.\n",
      "You probably need to get an updated matplotlibrc file from\n",
      "https://github.com/matplotlib/matplotlib/blob/v3.1.3/matplotlibrc.template\n",
      "or from the matplotlib source distribution\n"
     ]
    }
   ],
   "source": [
    "%pylab inline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import networkx as nx"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import tensorflow as tf\n",
    "sess = tf.compat.v1.InteractiveSession()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch \n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy import spatial\n",
    "from scipy.spatial import cKDTree\n",
    "import scipy.sparse as sp"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "from numpy import linalg as LA\n",
    "from numpy.linalg import inv\n",
    "from sklearn.decomposition import PCA\n",
    "import pandas as pd\n",
    "\n",
    "import cmath"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "class tictoc():\n",
    "    def __init__(self):\n",
    "        self.prev = 0\n",
    "        self.now = 0\n",
    "    def tic(self):\n",
    "        self.prev = time.time()\n",
    "    def toc(self):\n",
    "        self.now = time.time()\n",
    "        print( \"dt(s) = %.3g\" %(self.now - self.prev))\n",
    "        t = self.now - self.prev\n",
    "        self.prev = time.time()\n",
    "        return t\n",
    "        \n",
    "tt = tictoc()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 189,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Circular network\n",
    "i = 500\n",
    "Adj = eye(i,i,1)+eye(i,i,-1) \n",
    "Adj[0][-1] = 1\n",
    "Adj[-1][0] = 1\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 190,
   "metadata": {},
   "outputs": [],
   "source": [
    "#generate a graph from the adjacency matrix\n",
    "G = nx.from_numpy_matrix(Adj)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 156,
   "metadata": {},
   "outputs": [],
   "source": [
    "G = nx.grid_2d_graph(250, 250, periodic = True)\n",
    "\n",
    "\n",
    "#relable the graph and use one integers as an id\n",
    "label = ([ i[0] for i in list(G.nodes(data=True))])\n",
    "\n",
    "mapping = dict([ [label[i], i] for i in range(len(label))])\n",
    "G = nx.relabel_nodes(G, mapping)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 157,
   "metadata": {},
   "outputs": [],
   "source": [
    "##G= nx.barabasi_albert_graph(1000, 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 193,
   "metadata": {},
   "outputs": [],
   "source": [
    "def sparse_mx_to_torch_sparse_tensor(sparse_mx):\n",
    "    \"\"\"Convert a scipy sparse matrix to a torch sparse tensor.\"\"\"\n",
    "    sparse_mx = sparse_mx.tocoo().astype(np.float32)\n",
    "    indices = torch.from_numpy(\n",
    "        np.vstack((sparse_mx.row, sparse_mx.col)).astype(np.int64))\n",
    "    values = torch.from_numpy(sparse_mx.data)\n",
    "    shape = torch.Size(sparse_mx.shape)\n",
    "    return torch.sparse.FloatTensor(indices, values, shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 194,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "500\n"
     ]
    }
   ],
   "source": [
    "#adjacency mtx\n",
    "\n",
    "\n",
    "A = nx.to_numpy_matrix(G)\n",
    "#A = matrix(A)\n",
    "\n",
    "\n",
    "N = len(A)\n",
    "\n",
    "print(N)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 195,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "500"
      ]
     },
     "execution_count": 195,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(A)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 196,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1000"
      ]
     },
     "execution_count": 196,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#first neighbors\n",
    "#adjacency_list = torch.where(torch.triu(torch.tensor(A)))\n",
    "adjacency_list = torch.where((torch.tensor(A)))\n",
    "len(adjacency_list[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 200,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "2000"
      ]
     },
     "execution_count": 200,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#second neighbors\n",
    "sa = []\n",
    "\n",
    "for i in G.nodes(data= True):\n",
    "    for j in G.neighbors(i[0]):\n",
    "        for k in G.neighbors(j):\n",
    "            if i[0] != j and k != j: #i[0] !=k\n",
    "                sa += [[i[0], j, k]]\n",
    "                \n",
    "sa = array(sa)\n",
    "\n",
    "len(sa)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 201,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "2000"
      ]
     },
     "execution_count": 201,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#first neighbor impact to other first neighbor\n",
    "fa = []\n",
    "\n",
    "for i in G.nodes(data= True):\n",
    "    for j in G.neighbors(i[0]):\n",
    "        for k in G.neighbors(i[0]):\n",
    "            if j != i[0] and k!= i[0]: #j != k:\n",
    "                fa += [[i[0], j, k]]\n",
    "                \n",
    "fa = array(fa)\n",
    "\n",
    "len(fa)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 202,
   "metadata": {},
   "outputs": [],
   "source": [
    "fadj = ( torch.tensor(fa[:,0]), torch.tensor(fa[:,1]), torch.tensor(fa[:,2]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 203,
   "metadata": {},
   "outputs": [],
   "source": [
    "sadj = ( torch.tensor(sa[:,0]), torch.tensor(sa[:,1]), torch.tensor(sa[:,2]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 204,
   "metadata": {},
   "outputs": [],
   "source": [
    "#sparse\n",
    "A = sp.coo_matrix(A)\n",
    "\n",
    "A_norm = A + sp.eye(N)\n",
    "D_norm = sp.diags((1/np.sqrt(A_norm.sum(0))).tolist()[0])\n",
    "D_norm = D_norm.tocsr()\n",
    "DAD = D_norm.dot(A_norm.dot(D_norm))\n",
    "DAD = DAD.tocoo()\n",
    "\n",
    "DAD = sparse_mx_to_torch_sparse_tensor(DAD)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 207,
   "metadata": {},
   "outputs": [],
   "source": [
    "#input, output dimensions\n",
    "dim = 1\n",
    "#x = torch.eye(N) #.to(device)\n",
    "x = sp.eye(N)\n",
    "x = x.tocoo()\n",
    "x = sparse_mx_to_torch_sparse_tensor(x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 208,
   "metadata": {},
   "outputs": [],
   "source": [
    "class GCN(nn.Module):\n",
    "    def __init__(self, input_dim, output_dim, adj_mx):\n",
    "        super(GCN, self).__init__()\n",
    "        self.input_dim = input_dim\n",
    "        self.adj_mx = adj_mx\n",
    "        self.output_dim = output_dim\n",
    "        #self.dense = nn.Linear(input_dim, output_dim,bias=False)\n",
    "        self.weight = torch.nn.Parameter(torch.nn.init.uniform_(torch.FloatTensor(self.input_dim, self.output_dim), a=0.0, b=2*pi )) \n",
    "    #def reset_parameters(self):\n",
    "    #    stdv = 1. / math.sqrt(self.weight.size(1))\n",
    "    #    self.weight.data.uniform_(-stdv, stdv)\n",
    "        \n",
    "    def forward(self, input):\n",
    "        support = torch.spmm(input, self.weight)\n",
    "        output = torch.spmm(self.adj_mx, support)\n",
    "        \n",
    "        return output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 209,
   "metadata": {},
   "outputs": [],
   "source": [
    "class NeuralSync_Sparse(nn.Module):\n",
    "    def __init__(self, num_nodes, output_dim, hidden_dim_1, hidden_dim_2, hidden_dim_3,adj_mtx):\n",
    "        super(NeuralSync_Sparse, self).__init__()\n",
    "        self.num_nodes = num_nodes\n",
    "        self.output_dim = output_dim\n",
    "        self.hidden_dim_1 = hidden_dim_1\n",
    "        self.hidden_dim_2 = hidden_dim_2\n",
    "        self.hidden_dim_3 = hidden_dim_3\n",
    "        self.adj_mtx = adj_mtx\n",
    "        \n",
    "        self.weight1 = torch.nn.Parameter(torch.nn.init.uniform_(torch.FloatTensor(self.num_nodes, self.hidden_dim_1), a=0.0, b=2*pi ))\n",
    "        \n",
    "        self.GCN1 = GCN(self.hidden_dim_1, self.output_dim,self.adj_mtx.float())\n",
    "        \n",
    "        self.LR = nn.LeakyReLU()\n",
    "        self.Sigmoid = nn.Sigmoid()\n",
    "        \n",
    " \n",
    "    def forward(self, inp):\n",
    "        x = torch.spmm(inp, self.weight1)\n",
    "        \n",
    "        gnn1 = self.GCN1(x)\n",
    "        \n",
    "        gnn1 %= (2*pi)\n",
    "  \n",
    "        output = self.LR(gnn1)\n",
    "    \n",
    "        #output = self.Sigmoid(gnn1)*(2*pi)\n",
    "    \n",
    "        return  output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 210,
   "metadata": {},
   "outputs": [],
   "source": [
    "class LinearSync_Sparse(nn.Module):\n",
    "    def __init__(self, num_nodes, output_dim):\n",
    "        super(LinearSync_Sparse, self).__init__()\n",
    "        self.num_nodes = num_nodes\n",
    "        self.output_dim = output_dim\n",
    "        \n",
    "        #self.weight1 = torch.nn.Parameter(torch.nn.init.uniform_(torch.FloatTensor(self.num_nodes, self.output_dim), a=0.0, b=2*pi ))\n",
    "        self.weight1 = torch.nn.Parameter(torch.nn.init.uniform_(torch.FloatTensor(self.num_nodes, self.output_dim), a=0.0, b=2*pi ))\n",
    "        \n",
    "        \n",
    "        self.Sigmoid = nn.Sigmoid()\n",
    "        self.Tanh = nn.Tanh()\n",
    " \n",
    "    def forward(self, inp):\n",
    "        x = torch.spmm(inp, self.weight1)\n",
    "        \n",
    "        output = x#self.Tanh(x)*(2*pi) #self.Sigmoid(x)*(2*pi)\n",
    "    \n",
    "        return  output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 211,
   "metadata": {},
   "outputs": [],
   "source": [
    "#initialize phases between 0 and 2pi\n",
    "def init_weights(m):\n",
    "    if type(m) == nn.Linear:\n",
    "        torch.nn.init.uniform_(m.weight, a=0.0, b=2*pi)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 212,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Kuramoto-model\n",
    "def custom_loss_K(output, l):\n",
    "    \n",
    "    X = output\n",
    "    \n",
    "    X1 = X[adjacency_list[0]] #torch.index_select(input =X, dim = 0, index = adjacency_list[0], out=None)\n",
    "    X2 = X[adjacency_list[1]] #torch.index_select(input =X, dim = 0, index = adjacency_list[1], out=None)\n",
    "        \n",
    "    #V = (-1/2)*torch.sum( torch.cos(X1-X2), axis = 0)\n",
    "    V = -torch.sum( l*torch.cos(X2-X1), axis = 0)\n",
    "    \n",
    "    return V"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 92,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Hopf-Kuramoto model\n",
    "\n",
    "\n",
    "\n",
    "def custom_loss_KH(output, C, S1, S2):\n",
    "    \n",
    "    X = output\n",
    "    \n",
    "    XX1 = X[adjacency_list[0]]\n",
    "    XX2 = X[adjacency_list[1]]\n",
    "    \n",
    "    X1 = X[sadj[0]] #torch.index_select(input =X, dim = 0, index = adjacency_list[0], out=None)\n",
    "    X2 = X[sadj[1]] #torch.index_select(input =X, dim = 0, index = adjacency_list[1], out=None)\n",
    "    X3 = X[sadj[2]] #torch.index_select(input =X, dim = 0, index = adjacency_list[1], out=None)\n",
    "        \n",
    "    XXX1 = X[fadj[0]] #torch.index_select(input =X, dim = 0, index = adjacency_list[0], out=None)\n",
    "    XXX2 = X[fadj[1]] #torch.index_select(input =X, dim = 0, index = adjacency_list[1], out=None)\n",
    "    XXX3 = X[fadj[2]]    \n",
    "        \n",
    "    V = -( C*torch.sum( torch.cos(XX2-XX1), axis = 0)  + S1*torch.sum( torch.sin(XX2-XX1), axis = 0) + S2*( torch.sum( torch.sin(2*X2-X3-X1) - torch.sin(X3-X1)  , axis = 0) + torch.sum(torch.sin(XXX3+XXX2-2*XXX1), axis = 0) ))\n",
    "    #V = -(-C*torch.sum( torch.sin(XX2-XX1), axis = 0)  + S1*torch.sum( torch.cos(XX2-XX1), axis = 0) + S2*(torch.sum( torch.cos(2*X2-X3-X1) - torch.cos(X3-X1)  , axis = 0) + (1/2)*torch.sum(torch.cos(XXX3+XXX2-2*XXX1), axis = 0) ))\n",
    "    \n",
    "    \n",
    "    return V"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 176,
   "metadata": {},
   "outputs": [],
   "source": [
    "#stopping\n",
    "def difference(r):\n",
    "    return abs((max(r)-min(r))/max(r))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 100,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/csababoth/anaconda3/envs/tf1/lib/python3.7/site-packages/ipykernel_launcher.py:3: RuntimeWarning: divide by zero encountered in double_scalars\n",
      "  This is separate from the ipykernel package so we can avoid doing imports until\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "7.850599326528486e-08\n",
      "dt(s) = 3.13\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x720 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "Lin(10, '1', lr = 0.01)\n",
    "#Lin(1, 5, .1, '5_0.1_sparse-positive-lin', lr = .001)\n",
    "#Lin(1, 1, .1, '1_0.1_sparse-positive-lin', lr = .01)\n",
    "#Lin(1, .6, .3, '0.6_0.3_sparse-positive-lin', lr = .01)\n",
    "#Lin(1, 5, 12, '5_12_sparse-positive-lin', lr = .01)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 104,
   "metadata": {},
   "outputs": [],
   "source": [
    "#global order parameter\n",
    "\n",
    "s = 0\n",
    "for i in d:\n",
    "    s+= cmath.exp(complex(0, i)) \n",
    "    \n",
    "abs(s)/len(d)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 180,
   "metadata": {},
   "outputs": [],
   "source": [
    "def Lin(l, s_name, lr):\n",
    "    time_hist = []\n",
    "    state_history_lin = []\n",
    "\n",
    "    patience = 10\n",
    "    r = list(np.zeros(patience))\n",
    "\n",
    "\n",
    "    for i in range(1):\n",
    "        #x = torch.eye(N)\n",
    "    \n",
    "        #net = nn.Linear(N, dim, bias=False)\n",
    "        net = LinearSync_Sparse(N,dim)\n",
    "        net.apply(init_weights)\n",
    "\n",
    "        optimizer = torch.optim.Adam(net.parameters(), lr=lr)\n",
    "    \n",
    "        criterion = custom_loss_K#H\n",
    "        loss_history_lin= [] \n",
    "    \n",
    "        tt.tic()\n",
    "        for epoch in range(200000):  \n",
    "            inp = x\n",
    "        \n",
    "    \n",
    "            optimizer.zero_grad()\n",
    "\n",
    "            outputs = net(inp)\n",
    "        \n",
    "            state_history_lin += [outputs]\n",
    "        \n",
    "            \n",
    "            loss = criterion(outputs, l)#C,  S1, S2)\n",
    "\n",
    "    \n",
    "            loss.backward(retain_graph=True)\n",
    "            optimizer.step()\n",
    "        \n",
    "            loss_history_lin.append(loss.item())\n",
    "        \n",
    "        \n",
    "            r.append(loss.item())\n",
    "            r.pop(0)\n",
    "        \n",
    "       \n",
    "            if (difference(r)) < 10e-8:\n",
    "                print(difference(r))\n",
    "                break\n",
    "                \n",
    "    \n",
    "        \n",
    "        time_hist += [tt.toc()]\n",
    "        \n",
    "        d = pd.DataFrame(time_hist)\n",
    "        d.to_csv('./Result11_K/250x250-linear-%s-time.csv' %(s_name), header=True,index=False)\n",
    "\n",
    "        d = pd.DataFrame(loss_history_lin)\n",
    "        d.to_csv('./Result11_K/250x250-linear-%s-loss.csv' %(s_name), header=True,index=False)\n",
    "        \n",
    "        d = pd.DataFrame(state_history_lin[-1])\n",
    "        d.to_csv('./Result11_K/250x250-linear-%s-pattern.csv' %(s_name), header=True,index=False)\n",
    "        \n",
    "        \n",
    "        c = array([ round(double(i), 5) for i in state_history_lin[-1].detach()])\n",
    "        \n",
    "        plt.figure(figsize = (10,10))\n",
    "        plt.imshow( ((c.reshape(250,250)%(2*pi) )), cmap = 'hsv');\n",
    "        colorbar();\n",
    "        plt.savefig('./Result11_K/250x250-linear-%s.svg' %(s_name));"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 181,
   "metadata": {},
   "outputs": [],
   "source": [
    "def Neural(l, s_name,lr):\n",
    "    time_hist = []\n",
    "    state_history = []\n",
    "\n",
    "\n",
    "    patience = 10\n",
    "    r = list(np.zeros(patience))\n",
    "\n",
    "    for i in range(1):\n",
    "        #x = torch.eye(N)\n",
    "    \n",
    "        net = NeuralSync_Sparse(num_nodes=N, output_dim=dim, hidden_dim_1=10, hidden_dim_2=10, hidden_dim_3=5, adj_mtx= torch.tensor(DAD))\n",
    "        #net = NeuralSync(num_nodes=N, output_dim=dim, hidden_dim_1=10, hidden_dim_2=10, hidden_dim_3=5, adj_mtx= torch.tensor(DAD))\n",
    "    \n",
    "    \n",
    "        net.apply(init_weights)\n",
    "\n",
    "\n",
    "        optimizer = torch.optim.Adam(net.parameters(), lr=lr)\n",
    "    \n",
    "        criterion = custom_loss_K#H\n",
    "        loss_history_gcn= [] \n",
    "    \n",
    "        tt.tic()\n",
    "        for epoch in range(100):  \n",
    "            inp = x\n",
    "        \n",
    "    \n",
    "            optimizer.zero_grad()\n",
    "\n",
    "            outputs = net(inp)\n",
    "        \n",
    "            state_history += [outputs]\n",
    "        \n",
    "            #print(outputs)\n",
    "\n",
    "            loss = criterion(outputs, l) #C, S1, S2)\n",
    "\n",
    "    \n",
    "            loss.backward(retain_graph=True)\n",
    "            optimizer.step()\n",
    "        \n",
    "            loss_history_gcn.append(loss.item())\n",
    "        \n",
    "            r.append(loss.item())\n",
    "            r.pop(0)\n",
    "            \n",
    "            \n",
    "            #if (difference(r)) < 10e-10:\n",
    "            #    print(difference(r))\n",
    "            #    break\n",
    "    \n",
    "        net1 = nn.Linear(N, dim, bias=False)\n",
    "        net1.weight = torch.nn.Parameter(outputs.t().detach())\n",
    "        optimizer1 = torch.optim.Adam(net1.parameters(), lr=lr)\n",
    "    \n",
    "    \n",
    "        for epoch1 in range(300000):\n",
    "            inp = x\n",
    "    \n",
    "            optimizer1.zero_grad()\n",
    "            outputs1 = net1(inp)\n",
    "        \n",
    "            state_history += [outputs1]\n",
    "        \n",
    "            loss1 = criterion(outputs1, l) #C, S1, S2)\n",
    "\n",
    "            loss1.backward(retain_graph=True)\n",
    "            optimizer1.step()\n",
    "        \n",
    "            loss_history_gcn.append(loss1.item())\n",
    "        \n",
    "            r.append(loss1.item())\n",
    "            r.pop(0)\n",
    "        \n",
    "       \n",
    "            if (difference(r)) < 10e-8:\n",
    "                print(difference(r))\n",
    "                break\n",
    "    \n",
    "        \n",
    "        print('Finished training neural: ', i)\n",
    "    \n",
    "\n",
    "        time_hist += [tt.toc()]\n",
    "        \n",
    "        d = pd.DataFrame(time_hist)\n",
    "        d.to_csv('./Result11_K/250x250-neuralsync-%s-time.csv' %(s_name), header=True,index=False)\n",
    "\n",
    "        d = pd.DataFrame(loss_history_gcn)\n",
    "        d.to_csv('./Result11_K/250x250-neuralsync-%s-loss.csv' %(s_name), header=True,index=False)\n",
    "        \n",
    "        d = pd.DataFrame(state_history[-1])\n",
    "        d.to_csv('./Result11_K/250x250-neuralsync-%s-pattern.csv' %(s_name), header=True,index=False)\n",
    "        \n",
    "        \n",
    "        c = array([ round(double(i), 5) for i in state_history[-1].detach()])\n",
    "        \n",
    "        plt.figure(figsize = (10,10))\n",
    "        plt.imshow( ((c.reshape(250,250)%(2*pi) )), cmap = 'hsv');\n",
    "        plt.colorbar();\n",
    "        \n",
    "        plt.savefig('./Result11_K/250x250-neuralsync-%s.svg' %(s_name));"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "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.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
