{
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "w6_RqEhomxX1"
      },
      "outputs": [],
      "source": [
        "fo\n",
        "import numpy as np\n",
        "from scipy import linalg\n",
        "import matplotlib.pyplot as plt\n",
        "from matplotlib import colors"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "3kYnGwZZnPkU"
      },
      "outputs": [],
      "source": [
        "\n",
        "class Node:\n",
        "  def __init__(self, index = None):\n",
        "    self.L = index[0]\n",
        "    self.M = index[1]\n",
        "    self.R = index[2]\n",
        "    self.is_leaf = (self.R == self.L+1)\n",
        "\n",
        "  def left_child(self):\n",
        "    if self.is_leaf:\n",
        "        return self\n",
        "    else:\n",
        "        return Node([self.L,(self.L+self.M)//2,self.M])\n",
        "  def right_child(self):\n",
        "    if self.is_leaf:\n",
        "        return self\n",
        "    else :\n",
        "        return Node([self.M,(self.M+self.R)//2,self.R])\n",
        "  def test(self,matrix,k,T,sigma,pi):\n",
        "    Mlr = matrix[pi[self.L]][pi[self.R]]+np.mean(np.random.normal(size = T//3, scale = sigma))\n",
        "    Mkr = matrix[k][pi[self.R]]+np.mean(np.random.normal(size = T//3, scale = sigma))\n",
        "    Mkl = matrix[k][pi[self.L]]+np.mean(np.random.normal(size = T//3, scale = sigma))\n",
        "    if float(Mlr)  < float(np.min((Mkr,Mkl))):\n",
        "      return 0\n",
        "    elif float(Mkr) > float(Mkl):\n",
        "      return 1\n",
        "    else:\n",
        "      return -1\n",
        "  def value(self):\n",
        "    return [self.L,self.M,self.R]\n",
        "\n",
        "def binse(T,matrix,sigma,k,pi): #backtracking and binary search\n",
        "    T_1 = round(6*max(1,np.log2(k)))\n",
        "    T_2 = T//(T_1)\n",
        "    hist = [Node([0,(k-1)//2,k-1])]\n",
        "    for i in range(T_1):\n",
        "      node = hist[-1]\n",
        "      if (node.test(matrix,k,T_2//2,sigma,pi)  != 0):\n",
        "        if len(hist)>1:\n",
        "            hist.pop()\n",
        "      elif(node.right_child().test(matrix,k,T_2//2,sigma,pi)  == 0):\n",
        "        hist.append(node.right_child())\n",
        "      else:\n",
        "        hist.append(node.left_child())\n",
        "    return np.insert(pi,node.R,k)\n",
        "def naive(T,matrix,sigma,k,pi):\n",
        "    T_1 = round(max(1,np.log2(k)+2))\n",
        "    T_2 = T//(T_1)\n",
        "    hist = [Node([0,(k-1)//2,k-1])]\n",
        "    for i in range(T_1):\n",
        "      node = hist[-1]\n",
        "      if(node.right_child().test(matrix,k,T_2//2,sigma,pi)  == 0):\n",
        "        hist.append(node.right_child())\n",
        "      else:\n",
        "        hist.append(node.left_child())\n",
        "    return np.insert(pi,node.R,k)\n",
        "\n",
        "def ASIIn(T,matrix,sigma,pi):\n",
        "  n = np.shape(matrix)[0]\n",
        "  pmatrix = np.zeros((n,n))\n",
        "  for i in range(n):\n",
        "    for j in range(n):\n",
        "      pmatrix[i][j] = matrix[pi[i]][pi[j]]\n",
        "  hpi = np.array((0,1))\n",
        "  for k in range(2,n):\n",
        "    node = Node([0,(k-1)//2,k-1])\n",
        "    res = node.test(pmatrix,k,T//(2*n),sigma,hpi)\n",
        "    if res == 0:\n",
        "      hpi = naive(T//(2*n),pmatrix,sigma,k,hpi)\n",
        "    elif res == 1:\n",
        "      hpi = np.insert(hpi,k,k)\n",
        "    else:\n",
        "      hpi = np.insert(hpi,0,k)\n",
        "  return hpi\n",
        "\n",
        "def ASII(T,matrix,sigma,pi):\n",
        "  n = np.shape(matrix)[0]\n",
        "  pmatrix = np.zeros((n,n))\n",
        "  for i in range(n):\n",
        "    for j in range(n):\n",
        "      pmatrix[i][j] = matrix[pi[i]][pi[j]]\n",
        "  hpi = np.array((0,1))\n",
        "  for k in range(2,n):\n",
        "    node = Node([0,(k-1)//2,k-1])\n",
        "    res = node.test(pmatrix,k,T//(2*n),sigma,hpi)\n",
        "    if res == 0:\n",
        "      hpi = binse(T//(2*n),pmatrix,sigma,k,hpi)\n",
        "    elif res == 1:\n",
        "      hpi = np.insert(hpi,k,k)\n",
        "    else:\n",
        "      hpi = np.insert(hpi,0,k)\n",
        "  return hpi\n",
        "\n",
        "def AS(T,matrix,sigma,pi): #Adaptive sorting\n",
        "  n = np.shape(matrix)[0]\n",
        "  pmatrix = np.zeros((n,n))\n",
        "  for i in range(n):\n",
        "    for j in range(n):\n",
        "      pmatrix[i][j] = matrix[pi[i]][pi[j]]\n",
        "  pmatrix = pmatrix + np.random.normal(size = (n,n),scale = (sigma*n)/np.sqrt(T))\n",
        "  hpi = [np.argmin([np.sum(pmatrix[i,]) for i in range(n)])]\n",
        "  indices = [x for x in range(n)]\n",
        "  indices.remove(indices[hpi[0]])\n",
        "  for i in range(1,n):\n",
        "    ii = np.argmin([sum(abs(pmatrix[hpi[-1],] - pmatrix[j,])) for j in indices])\n",
        "    hpi.append(indices[ii])\n",
        "    indices.remove(indices[ii])\n",
        "  return hpi\n",
        "\n",
        "\n",
        "def inv(pi):\n",
        "  ipi = np.zeros(len(pi))\n",
        "  for i in range(len(pi)):\n",
        "    ipi[i] = int(np.where(pi == i)[0][0])\n",
        "  return [int(i) for i in ipi]\n",
        "\n",
        "\n",
        "def matrixgen(Delta,type):\n",
        "  matrix = np.zeros((n,n))\n",
        "  if type==1:\n",
        "    for i in range(n):\n",
        "          for j in range(n):\n",
        "            matrix[i][j] = (n -abs(j - i))*Delta\n",
        "\n",
        "  if type==2: #none toeplitz\n",
        "      mods = np.linspace(1,10,n)\n",
        "      for i in range(n):\n",
        "            for j in range(n):\n",
        "              if j >= i:\n",
        "                matrix[i][j] = (n -abs(j - i))*max(i,n-j-1)**1.5*Delta\n",
        "                matrix[j][i] =  matrix[i][j]\n",
        "  if type==3: #none toeplitz\n",
        "      for i in range(n):\n",
        "            for j in range(n):\n",
        "              if abs(i-j) < n//4:\n",
        "                if j >= i:\n",
        "                  matrix[i][j] = (n -abs(j - i))*max(i,n-j-1)*(Delta+10)\n",
        "                  matrix[j][i] =  matrix[i][j]\n",
        "              else:\n",
        "                if j >= i:\n",
        "                  matrix[i][j] = (n -abs(j - i))*max(i,n-j-1)*Delta\n",
        "                  matrix[j][i] =  matrix[i][j]\n",
        "\n",
        "  if type==4: #none toeplitz\n",
        "    mods = np.random.uniform(1,10,n)\n",
        "    np.fill_diagonal(matrix, mods)\n",
        "    for k in range(1,n):\n",
        "      for i in range(k,n):\n",
        "        matrix[i,i-k] = min(matrix[i-1,i-k],matrix[i,i-k+1]) - np.random.uniform(Delta,10*Delta,1)\n",
        "        matrix[i-k,i] = matrix[i,i-k]\n",
        "\n",
        "\n",
        "\n",
        "def spec(T,matrix,sigma,pi):\n",
        "  n = np.shape(matrix)[0]\n",
        "  pmatrix = np.zeros((n,n))\n",
        "  for i in range(n):\n",
        "    for j in range(n):\n",
        "      pmatrix[i][j] = matrix[pi[i]][pi[j]]\n",
        "  pmatrix = pmatrix + np.random.normal(size = (n,n),scale = sigma)\n",
        "\n",
        "  diags = [np.sum(pmatrix[i,:])for i in range(n)]\n",
        "  L = np.zeros((n,n)) - pmatrix\n",
        "  for i in range(n):\n",
        "    L[i,i] = diags[i]\n",
        "  eval, evec = np.linalg.eig(L)\n",
        "  w, vl, vr = linalg.eig(L, left=True)\n",
        "  hpi = vl[:,1].argsort()\n",
        "  return(hpi)\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "GfvFhdn8pyDU"
      },
      "outputs": [],
      "source": [
        "plt.plot()"
      ]
    }
  ],
  "metadata": {
    "colab": {
      "provenance": []
    },
    "kernelspec": {
      "display_name": "Python 3",
      "name": "python3"
    },
    "language_info": {
      "name": "python"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}