{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "a9ec5a5e-acd7-4f2c-a50f-cf2bf24f5c73",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import pickle\n",
    "import matplotlib.pyplot as plt\n",
    "from van_code.nn import NeuralVANMultilevel_block_wise\n",
    "from van_code.ising import ising_energy,analytical_solution,local_ising_energy\n",
    "from van_code.montecarlo import *\n",
    "from van_code.utils import *\n",
    "from van_code.obs import Obs\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5c4569e9-4b9e-4e87-871e-93d83c0511da",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "You are using device=mps.\n"
     ]
    }
   ],
   "source": [
    "if torch.backends.mps.is_available():\n",
    "    device = torch.device(\"mps\")\n",
    "    default_dtype_torch=torch.float32\n",
    "elif torch.cuda.is_available():\n",
    "    device = 'cuda'\n",
    "else:\n",
    "    device = torch.device(\"cpu\")\n",
    "print(f'You are using device={device}.')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8164be43-0322-4b8c-ae36-a9b0b0e20310",
   "metadata": {},
   "source": [
    "-------------\n",
    "-------------\n",
    "# Model and parameters\n",
    "Our multilevel architecture uses different blocks of Autoregressive neural networks which are based on the VAN architecture by [Wu et al. (2019)](https://journals.aps.org/prl/abstract/10.1103/PhysRevLett.122.080602)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "a7f18c2f-e4e7-4a4d-81ab-153f60db5425",
   "metadata": {},
   "outputs": [],
   "source": [
    "# This cell defines important parameters for the simulation\n",
    "\n",
    "Lc=2 # defines the coarser lattice\n",
    "beta=0.44 # defines the beta value (inverse of the temperature)\n",
    "\n",
    "# VAN parameters\n",
    "net_depth = 3\n",
    "net_width= 16\n",
    "half_kernel_size =6\n",
    "\n",
    "# CNN VAN parameters\n",
    "# The i-th element of the list correspond to the parameters associated to the i-th block in the multilevel\n",
    "\n",
    "hidden_size=[[16]] * 10 # Hidden size of each CondVAN\n",
    "kernel_size=[5] *10  # Kernel of CondVAN\n",
    "\n",
    "# @ELIA what is the difference between VAN above, the CondVAN, and CNN VAN? We should make it clear!\n",
    "\n",
    "nlevels = 2 # defines the numbe rof levels in the multilevel. Every level doubles the lattice. Example: if the coarser is 2x2 after 3 levels we have 16x16\n",
    "hb_last = True # whether to use heatbath to sample unbiased configuration from the last ARNN\n",
    "\n",
    "pretrained = True  # wheter to initilize j block with j-1 block's parameters @ELIA Can you explain this better?\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "e6e338fe-3d49-412c-99be-d84c4f7d2cf6",
   "metadata": {},
   "outputs": [],
   "source": [
    "van_hyp={\n",
    "    'net_depth':net_depth,\n",
    "    'net_width': net_width,\n",
    "    'half_kernel_size':half_kernel_size,\n",
    "    'bias':False,'z2':False,\n",
    "    'res_block':True,\n",
    "    'x_hat_clip':False,\n",
    "    'final_conv':True,\n",
    "    'epsilon':1.e-8,\n",
    "    'device':device\n",
    "}\n",
    "\n",
    "block_net_hyp={\n",
    "    'hidden_size':hidden_size,\n",
    "    'kernel_size':kernel_size,\n",
    "    'epsilon':1e-7,\n",
    "    'level':0,\n",
    "    'device':device\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "994f246c-3856-4c65-8e2d-106364f5bb4e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "=======================================================================================================================================================\n",
      "\n",
      "You will start from coarser lattices of shape 2x2 and sample finer lattices of resolution 16x16 using 3 multilevel steps.\n",
      "\n",
      "=======================================================================================================================================================\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# Building the model\n",
    "\n",
    "model = NeuralVANMultilevel_block_wise(\n",
    "    Lc,\n",
    "    van_hyp,\n",
    "    block_net_hyp,\n",
    "    nlevels,\n",
    "    hb_last,\n",
    "    ising_energy,\n",
    "    local_ising_energy,\n",
    "    beta,\n",
    "    device\n",
    ")\n",
    "Lf = model.Lf\n",
    "#print(model)\n",
    "\n",
    "print(f'\\n\\n=======================================================================================================================================================\\n')\n",
    "print(f'You will start from coarser lattices of shape {Lc}x{Lc} and sample finer lattices of resolution {model.Lf}x{model.Lf} using {nlevels} multilevel steps.')\n",
    "print(f'\\n=======================================================================================================================================================\\n')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "6989f290-fcf4-42e6-8914-e7b658d58175",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define default parameters for optimizer and scheduler\n",
    "\n",
    "optimizer = torch.optim.Adam(model.parameters(), lr=0.0005)\n",
    "scheduler = torch.optim.lr_scheduler.ReduceLROnPlateau(optimizer, 'min', factor=0.92, patience=1000, min_lr=1e-07)\n",
    "\n",
    "# Define additional training parameters. Arbitrary values can be chosen\n",
    "# the values chosen below are optimized for the purpose of this demo.\n",
    "\n",
    "bs = [16] * (nlevels+1)\n",
    "nepochs = 2 * [500] * nlevels + [500]\n",
    "print_freq = 10\n",
    "lr = 2 * [0.001] + [0.001] * (nlevels-1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "6d9a949d-4642-4a59-a987-2282d0a12854",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Choose main path to store model and logs\n",
    "main_path=''+'../data/'\n",
    "\n",
    "# Creates model identifier with corresponding params\n",
    "# '/Lf'+'_beta'+'_nblocks'+'_PCNNdepth'+'_width'+'_half_ker'+'_CCNNhs'+'_ks'\n",
    "\n",
    "model_id = f'{str(model.Lf)}_{str(beta)}_{str(nlevels)}_{str(net_depth)}_{str(net_width)}_{str(half_kernel_size)}_{str(hidden_size[0][0])}_{str(kernel_size[0])}'\n",
    "\n",
    "lc_path = main_path + 'training/' + model_id\n",
    "h_path = lc_path + '_history.log' # Path to Training history\n",
    "\n",
    "## Dict of paths to store different results\n",
    "\n",
    "res_paths = {\n",
    "    'weights': main_path+ 'model/'+ model_id +'.chckpnt', # models' weights\n",
    "    'sim' : main_path + 'results/' + model_id + '_measures.log',  # Simulations logging\n",
    "    'sim_md' : main_path + 'results/' + model_id + '_measures_modedrop.log',  # Simulations mode dropping\n",
    "    'cluster' : main_path + 'results/' + model_id + '_measuresCluster.log',  # Simulations\n",
    "    'dict_hist' : lc_path + '_historyDict.pkl',  # dictionary learning curve\n",
    "    'mh' : main_path+'results/'+ model_id +'_measuresIMH.log' # neural MCMC sampling results\n",
    "}\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "627aaeee-8aa4-41ee-a7ab-950d59d4edca",
   "metadata": {},
   "source": [
    "-------------\n",
    "-------------\n",
    "# Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "a2de395c-9d62-4310-a4e6-3b55f0028563",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "============================================================================================================================\n",
      "\n",
      "Nothing to load. Training of new model... \n",
      "\n",
      "============================================================================================================================\n",
      "\n",
      "Training VAN layers...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/kimnicoli/Projects/multilevelRG/venv/lib/python3.9/site-packages/torch/amp/grad_scaler.py:132: UserWarning: torch.cuda.amp.GradScaler is enabled, but CUDA is not available.  Disabling.\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "10 -0.0229491 -2.5482035 2.376858 0.12304192 -3.7982407 -2.7682035\n",
      "20 -0.015963137 -2.7724369 1.6171459 0.12522653 -3.850008 -2.7724369\n",
      "30 -0.01877281 -2.9840608 0.73225653 0.13679679 -3.8168726 -2.7640607\n",
      "40 -0.049151957 -2.9683857 0.6627169 0.15294091 -3.7019234 -2.7483857\n",
      "50 -0.38977233 -3.3656783 2.7595196 0.32160327 -4.5578513 -2.7056782\n",
      "60 -0.62615234 -3.351481 2.0331695 0.37008435 -4.2360463 -2.691481\n",
      "70 -0.6921334 -4.169017 0.99128306 0.6999538 -4.5342607 -2.1890168\n",
      "80 -0.4480962 -4.263093 0.4238952 0.87139076 -4.4057484 -1.8430932\n",
      "90 0.083502404 -4.44978 0.008630501 0.9890719 -4.4542694 -1.1497802\n",
      "100 -0.107916586 -4.3030653 0.025757305 0.9861484 -4.3131485 -1.0030649\n",
      "110 -0.053690936 -4.2897816 0.0050762957 0.9962781 -4.291977 -0.9897815\n",
      "120 -0.07405944 -4.265761 0.0051931995 0.9959133 -4.26806 -1.1857607\n",
      "130 1.315531e-05 -4.2818866 1.40810325e-05 0.99998665 -4.281893 -0.7618865\n",
      "140 4.8792936e-06 -4.277116 5.180469e-06 0.99999523 -4.277118 -0.7571157\n",
      "150 -0.041862957 -4.282188 0.0029708568 0.99769 -4.283498 -0.98218805\n",
      "160 -0.07187645 -4.3083515 0.00977962 0.9934687 -4.3124547 -1.0083512\n",
      "170 -0.21690759 -4.2542925 0.048362978 0.9721678 -4.2737317 -1.3942922\n",
      "180 0.03477328 -4.348195 0.0016364761 0.99823725 -4.348999 -1.0481954\n",
      "190 0.041196313 -4.3450766 0.033373084 0.96439445 -4.3612785 -1.2650766\n",
      "200 -0.24167833 -4.2650137 0.07802116 0.9620465 -4.294856 -1.1850134\n",
      "210 0.015389577 -4.385481 0.057689276 0.96117896 -4.4092855 -1.5254804\n",
      "220 0.054829847 -4.4229813 0.0038691298 0.99551255 -4.4249315 -1.122981\n",
      "230 0.016616492 -4.394974 0.012639627 0.98743486 -4.400993 -1.3149743\n",
      "240 -0.072123006 -4.3441253 0.03436559 0.9732049 -4.359253 -1.4841254\n",
      "250 -0.068230554 -4.352413 0.008540133 0.9942156 -4.3560133 -1.052413\n",
      "260 0.09658255 -4.411108 0.010947351 0.9853709 -4.416916 -1.1111085\n",
      "270 -0.073063344 -4.3697495 0.012040758 0.9923239 -4.3747277 -1.2897499\n",
      "280 -0.1131226 -4.3504534 0.016220916 0.98956066 -4.3571796 -1.2704533\n",
      "290 -0.06814846 -4.3483925 0.009817509 0.9934962 -4.352502 -1.2683928\n",
      "300 0.050633628 -4.391553 0.013747822 0.9834473 -4.3985643 -1.3115532\n",
      "310 -0.049154624 -4.38612 0.004234839 0.996815 -4.387966 -1.0861201\n",
      "320 0.02045837 -4.378669 0.034615606 0.9531317 -4.397154 -1.7386692\n",
      "330 -0.049719475 -4.362514 0.0022492479 0.9980821 -4.3635364 -1.2825141\n",
      "340 -0.100578666 -4.3559446 0.009763897 0.99271107 -4.360198 -1.275945\n",
      "350 -0.19810316 -4.3162584 0.023043603 0.9824836 -4.3263974 -1.6762581\n",
      "360 -0.019457374 -4.3779516 0.00068525015 0.9994022 -4.3782654 -1.0779519\n",
      "370 -0.08058365 -4.33382 0.005016837 0.9958003 -4.336088 -1.6938199\n",
      "380 0.4461374 -4.517736 0.09030023 0.8982486 -4.564543 -2.0977361\n",
      "390 -0.05026225 -4.3459535 0.03169637 0.9644294 -4.361762 -1.9259536\n",
      "400 -0.044158027 -4.379912 0.015458548 0.98676556 -4.3869333 -1.5199118\n",
      "410 -0.11651458 -4.3550687 0.009676388 0.9923892 -4.359357 -1.4950688\n",
      "420 -0.046388414 -4.367481 0.0021327243 0.9981887 -4.368449 -1.2874813\n",
      "430 -0.12014413 -4.336449 0.013661325 0.99039716 -4.3422785 -1.4764493\n",
      "440 0.20889486 -4.4429703 0.02887308 0.9576353 -4.4588976 -1.5829704\n",
      "450 0.10977253 -4.402495 0.035682518 0.9440824 -4.422671 -1.5424948\n",
      "460 0.0822309 -4.3901024 0.02454258 0.96670014 -4.4031878 -1.3101026\n",
      "470 -0.012413608 -4.375476 0.00039700954 0.99963766 -4.3756604 -1.0754762\n",
      "480 7.402222e-05 -4.392536 7.873461e-05 0.9999256 -4.3925734 -0.87253654\n",
      "490 0.00010191952 -4.3924885 0.000108549066 0.9998989 -4.3925395 -0.8724886\n",
      "500 0.020291988 -4.378042 0.0053032325 0.9944669 -4.380616 -1.2980418\n",
      "Time taken for this level now 9.360645055770874\n",
      "Unfreezing and training up to level 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/kimnicoli/Projects/multilevelRG/venv/lib/python3.9/site-packages/torch/_compile.py:31: UserWarning: optimizer contains a parameter group with duplicate parameters; in future, this will cause an error; see github.com/pytorch/pytorch/issues/40967 for more information\n",
      "  return disable_fn(*args, **kwargs)\n",
      "/Users/kimnicoli/Projects/multilevelRG/venv/lib/python3.9/site-packages/torch/amp/autocast_mode.py:265: UserWarning: User provided device_type of 'cuda', but CUDA is not available. Disabling\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "10 -2.9730525 -12.882533 8.800662 0.084409565 -17.05317 -9.362533\n",
      "20 -4.9976163 -13.940269 3.865408 0.30239648 -15.40251 -8.440269\n",
      "30 -1.2499275 -15.377146 0.37227473 0.85980254 -15.509138 -3.717146\n",
      "40 -0.35657665 -15.361813 0.03601429 0.9750871 -15.377081 -3.5918128\n",
      "50 -0.77516115 -15.276748 0.10442063 0.9442976 -15.317583 -4.276749\n",
      "60 0.7450986 -15.454123 0.1170413 0.74381614 -15.535208 -2.5841222\n",
      "70 0.20547679 -15.482303 0.08816128 0.8244671 -15.539112 -3.2723029\n",
      "80 -0.078419484 -15.463309 0.14387128 0.8614605 -15.531452 -4.0233088\n",
      "90 0.35053766 -15.475313 0.06707022 0.92319036 -15.509607 -4.475314\n",
      "100 -0.48531404 -15.474856 0.14870094 0.86090857 -15.546154 -4.8048553\n",
      "110 0.21144693 -15.521956 0.06412651 0.95188534 -15.549843 -4.5219564\n",
      "120 -1.2116967 -15.40955 0.19142452 0.95583695 -15.466385 -2.8695498\n",
      "130 -0.08923599 -15.454542 0.1102099 0.8937464 -15.5073185 -3.904543\n",
      "140 -0.48551553 -15.376119 0.04487321 0.9713375 -15.394718 -3.606119\n",
      "150 0.15584525 -15.503258 0.02024541 0.97328466 -15.513964 -2.7432578\n",
      "160 0.1891647 -15.4735775 0.11249363 0.9166108 -15.522253 -6.453578\n",
      "170 -0.17435086 -15.43906 0.04581969 0.96256965 -15.45974 -6.08906\n",
      "180 -0.004370369 -15.474438 0.015427176 0.98328227 -15.482033 -4.694439\n",
      "190 -0.46384746 -15.390476 0.071169645 0.9689209 -15.416481 -2.4104762\n",
      "200 -0.7283862 -15.339648 0.083146624 0.945469 -15.374632 -4.4496484\n",
      "210 0.099561386 -15.461448 0.018011546 0.97723943 -15.470821 -3.6914482\n",
      "220 0.0030114055 -15.502333 0.05535381 0.9258708 -15.531778 -4.172333\n",
      "230 -0.0446198 -15.518513 0.026777703 0.9706781 -15.531815 -3.8585136\n",
      "240 -0.051610366 -15.547724 0.025326125 0.97902673 -15.559149 -3.6677246\n",
      "250 0.24521634 -15.547734 0.066121764 0.9156357 -15.5828495 -4.4377346\n",
      "260 -0.23453401 -15.463939 0.04248088 0.96704507 -15.4824295 -4.4639387\n",
      "270 0.15186577 -15.50849 0.046486303 0.9332072 -15.533997 -3.1884894\n",
      "280 -0.31498042 -15.389828 0.039598387 0.9664586 -15.407816 -4.609828\n",
      "290 -0.051023304 -15.452223 0.06245239 0.95398647 -15.47925 -5.882222\n",
      "300 0.6101292 -15.488649 0.062312186 0.9192615 -15.522081 -3.828649\n",
      "310 -0.102633655 -15.488752 0.10578894 0.89912194 -15.539978 -6.138753\n",
      "320 -0.14063463 -15.486881 0.011250787 0.9904198 -15.491991 -4.266881\n",
      "330 0.031533472 -15.517868 0.004066364 0.9959323 -15.519815 -2.537868\n",
      "340 -0.29105267 -15.450436 0.06103667 0.9429153 -15.479221 -5.1104355\n",
      "350 -0.047742575 -15.513085 0.0861589 0.88886386 -15.559538 -4.953085\n",
      "360 -0.18798521 -15.3992405 0.0732818 0.93804336 -15.432352 -4.5092406\n",
      "370 0.09795524 -15.515585 0.009998189 0.9889587 -15.520552 -3.9655848\n",
      "380 -0.3880032 -15.396753 0.04158095 0.9747226 -15.413727 -4.286753\n",
      "390 -0.39586657 -15.428949 0.026051074 0.9758388 -15.441132 -4.2089496\n",
      "400 0.2093235 -15.551527 0.03431913 0.9637674 -15.568474 -4.3315277\n",
      "410 -0.9460965 -15.351901 0.07928248 0.95615053 -15.383404 -4.2419014\n",
      "420 -0.18253052 -15.42717 0.07477219 0.92867225 -15.462807 -5.637169\n",
      "430 -0.16876185 -15.428819 0.01995198 0.9806414 -15.438269 -3.438819\n",
      "440 0.2006937 -15.501061 0.019403256 0.9825136 -15.510019 -4.391062\n",
      "450 -0.23777679 -15.432541 0.017279305 0.9885363 -15.439772 -3.442541\n",
      "460 -0.3627087 -15.522251 0.06780259 0.925613 -15.556147 -3.8622506\n",
      "470 -0.73837733 -15.407061 0.09604619 0.9451661 -15.445564 -4.1870604\n",
      "480 0.047451302 -15.484669 0.04549126 0.94836605 -15.507648 -4.374669\n",
      "490 -0.13440587 -15.54055 0.0988161 0.8871261 -15.589687 -3.9905512\n",
      "500 0.7501695 -15.548257 0.120917104 0.8379875 -15.615303 -4.548257\n",
      "Time taken for this level now 31.57036018371582\n",
      "Unfreezing and training up to level 2\n",
      "10 3.256816 -59.597313 0.55876863 0.5331431 -59.90621 -12.187314\n",
      "20 -11.09506 -58.297955 8.209638 0.5903563 -59.146164 -13.637955\n",
      "30 0.6763725 -59.03583 0.7841248 0.3630509 -59.509293 -9.865832\n",
      "40 0.39427733 -59.033302 0.45079726 0.47497302 -59.310127 -13.163303\n",
      "50 -3.6835878 -59.136097 1.0303285 0.6453198 -59.46805 -14.2561\n",
      "60 0.45173752 -59.490467 0.35504678 0.6462658 -59.677475 -10.760466\n",
      "70 1.8398995 -59.74185 0.51514536 0.6133023 -59.991264 -14.091854\n",
      "80 -4.393445 -59.57147 0.7472229 0.62096745 -59.854958 -15.131469\n",
      "90 -1.3717644 -59.418262 0.6275707 0.6176458 -59.69023 -18.718266\n",
      "100 -3.1965337 -59.472538 0.7599345 0.72745925 -59.7274 -14.372538\n",
      "110 -1.6661823 -59.685463 0.8434055 0.59023017 -60.012093 -16.895462\n",
      "120 -1.0845742 -59.992798 0.5265073 0.71215993 -60.211296 -15.002798\n",
      "130 -2.7670577 -60.06385 0.69770974 0.578205 -60.361465 -16.393848\n",
      "140 -2.9515953 -59.891907 1.2834314 0.3266357 -60.485855 -22.711906\n",
      "150 -1.105489 -60.001167 0.62373215 0.6322494 -60.280212 -20.401165\n",
      "160 -4.939314 -59.585617 0.5454371 0.8111433 -59.77443 -18.00562\n",
      "170 -2.2965865 -59.614807 0.428335 0.7592364 -59.78012 -18.254807\n",
      "180 -1.0342944 -59.785664 0.56470555 0.6782373 -60.028275 -19.965664\n",
      "190 -4.273444 -59.926308 0.6755799 0.66276866 -60.20452 -20.546307\n",
      "200 -3.2147665 -59.706207 0.32590106 0.7255913 -59.86087 -16.036207\n",
      "210 1.1249539 -59.90819 0.23956384 0.79019797 -60.02471 -14.918191\n",
      "220 -1.7958236 -59.93872 0.33775333 0.7748196 -60.088387 -17.14872\n",
      "230 -3.219264 -59.871056 0.51705706 0.7643451 -60.064003 -15.871054\n",
      "240 -0.42522186 -59.87593 0.41300282 0.6893702 -60.06781 -16.31593\n",
      "250 -0.46619797 -59.86566 0.22186911 0.6805647 -59.99723 -18.835663\n",
      "260 0.53891945 -59.925453 0.3036243 0.5644491 -60.115776 -14.275453\n",
      "270 1.3315227 -59.943542 0.2059387 0.6319311 -60.08087 -15.063543\n",
      "280 -1.0484059 -59.701363 0.1339387 0.9265173 -59.754105 -14.931364\n",
      "290 -1.9494424 -59.790207 0.23220573 0.8140385 -59.899338 -20.740206\n",
      "300 3.1530771 -59.897583 0.2795093 0.6325582 -60.066055 -14.137585\n",
      "310 -1.287854 -60.10331 0.16562122 0.82888526 -60.184784 -15.443312\n",
      "320 0.34676027 -59.88965 0.26795852 0.8129152 -60.00306 -16.659649\n",
      "330 0.7654916 -59.953438 0.08956902 0.8891487 -60.000618 -14.96344\n",
      "340 -1.7351747 -59.736217 0.1655956 0.80879503 -59.82231 -18.706219\n",
      "350 -1.1930807 -60.024067 0.2244524 0.80738324 -60.130165 -17.674068\n",
      "360 0.22527921 -59.976048 0.12952127 0.8592583 -60.042755 -12.786047\n",
      "370 -0.4205675 -59.90781 0.15532939 0.87243015 -59.978806 -14.807812\n",
      "380 -0.87504864 -59.84575 0.27881536 0.82991034 -59.96003 -22.11575\n",
      "390 0.38202775 -60.026207 0.10726936 0.90555876 -60.07663 -17.566204\n",
      "400 1.0517194 -60.16628 0.12865676 0.88361025 -60.22811 -16.166279\n",
      "410 1.6109612 -60.166363 0.19002388 0.826555 -60.259933 -18.696365\n",
      "420 0.11453092 -60.02675 0.0801548 0.908479 -60.067818 -15.036746\n",
      "430 -1.3984632 -60.005882 0.20616889 0.835807 -60.100983 -21.06588\n",
      "440 -0.25446916 -59.945282 0.103483304 0.90505457 -59.994637 -16.495285\n",
      "450 -2.4394045 -59.975655 0.36148173 0.7477564 -60.13527 -22.025654\n",
      "460 0.1444099 -60.03503 0.1872407 0.8253892 -60.126778 -20.215029\n",
      "470 -1.7023268 -59.883526 0.24410371 0.7899448 -60.00004 -21.823524\n",
      "480 -2.3493826 -59.984303 0.262676 0.86103016 -60.088783 -18.404305\n",
      "490 -1.206562 -59.785614 0.18669146 0.8376775 -59.873222 -17.545614\n",
      "500 -3.3166394 -59.723522 0.15083371 0.9004537 -59.78724 -19.463522\n",
      "Time taken for this level now 100.80664300918579\n",
      "Unfreezing and training up to level 3\n",
      "10 10.088288 -238.06177 1.0284258 0.47237828 -238.57063 -71.851776\n",
      "20 -7.1786737 -237.8284 0.9045714 0.5014954 -238.25008 -73.0484\n",
      "30 -2.8006444 -237.56874 2.0128827 0.4256786 -238.34541 -73.77875\n",
      "40 -4.6528683 -237.70657 0.7599687 0.59381646 -238.0153 -77.76658\n",
      "50 10.470377 -237.95702 1.0808697 0.2472451 -238.59924 -71.08702\n",
      "60 -6.726845 -237.98778 0.8378416 0.5072521 -238.3527 -72.76778\n",
      "70 2.796534 -238.08185 0.61415535 0.58885384 -238.3815 -65.82185\n",
      "80 -7.747508 -238.0899 0.69319564 0.5513231 -238.4192 -81.449905\n",
      "90 7.026476 -238.11954 0.8964681 0.47185957 -238.56027 -77.73954\n",
      "100 -0.12225628 -238.1481 0.39843687 0.7242584 -238.33154 -65.77811\n",
      "110 0.55816936 -237.90141 0.23539439 0.85871434 -237.99985 -67.40142\n",
      "120 2.648716 -238.14725 0.60883814 0.60663825 -238.42229 -78.427246\n",
      "130 -0.5773144 -238.14117 0.27173495 0.859396 -238.2471 -77.10118\n",
      "140 -2.2115507 -238.30911 0.992555 0.5971421 -238.70062 -84.52911\n",
      "150 4.7850976 -238.33224 0.33797795 0.76128966 -238.48232 -68.38225\n",
      "160 2.745989 -237.89293 0.52126575 0.5986931 -238.14833 -78.06293\n",
      "170 -4.4597626 -237.75117 0.8365099 0.3971153 -238.20396 -74.29118\n",
      "180 -4.349105 -238.18059 0.36529085 0.80378777 -238.32547 -69.4406\n",
      "190 -10.567161 -238.03693 0.8963996 0.682499 -238.37126 -82.05693\n",
      "200 -2.1218233 -237.96584 0.22891745 0.8711996 -238.05893 -70.54584\n",
      "210 -2.9979591 -238.3958 0.39056638 0.762941 -238.5656 -71.1958\n",
      "220 -3.4939232 -238.15991 0.18286115 0.8258175 -238.2485 -77.77991\n",
      "230 8.860631 -238.53287 0.64712054 0.36510363 -238.92773 -67.04286\n",
      "240 -5.6339393 -238.14432 0.475054 0.56139123 -238.40259 -78.20433\n",
      "250 -3.120102 -237.7869 0.4305883 0.7267825 -237.97543 -71.576904\n",
      "260 -7.374172 -237.97084 0.6117946 0.71279055 -238.21582 -81.55084\n",
      "270 -1.3852587 -238.00935 0.3419541 0.79953086 -238.15154 -68.05934\n",
      "280 6.031864 -237.89682 0.17677379 0.83613867 -237.98323 -70.91681\n",
      "290 -2.524805 -238.28763 0.4868492 0.6705658 -238.50836 -74.05763\n",
      "300 -1.6133223 -238.2309 0.4743267 0.6171324 -238.46191 -76.3109\n",
      "310 2.6417155 -238.28749 0.45719036 0.6571947 -238.51047 -79.00749\n",
      "320 2.0688708 -238.20709 0.077245615 0.92812335 -238.24393 -62.75709\n",
      "330 1.4583404 -238.22137 0.2800753 0.69812983 -238.37193 -71.02137\n",
      "340 1.4107037 -238.05135 1.1195441 0.34861827 -238.70413 -69.75134\n",
      "350 2.0432496 -238.3391 0.32028195 0.6303614 -238.5204 -74.43909\n",
      "360 -8.736474 -237.85474 0.4623019 0.7146637 -238.05272 -71.09474\n",
      "370 -6.959962 -237.7113 0.6126715 0.5845567 -237.99837 -80.0813\n",
      "380 0.022214413 -238.08226 0.75025666 0.59110445 -238.40723 -79.13226\n",
      "390 -3.9858155 -238.35744 0.3387957 0.7980439 -238.49617 -69.50743\n",
      "400 -6.4363246 -238.14503 0.9316484 0.5107783 -238.5729 -76.99504\n",
      "410 -0.31422353 -237.9216 0.33357897 0.7268491 -238.07716 -83.5916\n",
      "420 -2.6297035 -238.14568 0.46070302 0.760013 -238.32433 -78.425674\n",
      "430 -5.1465893 -237.97516 0.38109386 0.7022324 -238.157 -77.92516\n",
      "440 1.8668454 -238.03961 0.6679797 0.527507 -238.37852 -64.129616\n",
      "450 -12.114132 -238.12567 1.1653204 0.42563963 -238.65605 -93.25567\n",
      "460 1.9755329 -238.22517 0.2829725 0.83820814 -238.34193 -71.79517\n",
      "470 2.223198 -238.0682 0.67593706 0.43638545 -238.44716 -68.5582\n",
      "480 -4.953652 -238.2727 0.37682578 0.7634534 -238.43884 -73.49271\n",
      "490 -2.5344262 -238.28198 0.34251526 0.7459501 -238.44083 -83.951996\n",
      "500 4.49362 -238.3367 0.24689983 0.8522918 -238.44083 -74.4367\n",
      "Total time taken : 335.13623690605164\n",
      "\n",
      "============================================================================================================================\n",
      "\n",
      "Training successful!\n",
      "\n",
      "============================================================================================================================\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# Load model if checkpoint exists, otherwise train a new model.\n",
    "\n",
    "if os.path.exists(res_paths['weights']):\n",
    "    print(f'\\n============================================================================================================================\\n')\n",
    "    print(f\"Loading model {res_paths['weights']}.\")\n",
    "    print(f'\\n============================================================================================================================\\n')\n",
    "    load(model, optimizer, res_paths['weights'])\n",
    "    with open(res_paths['dict_hist'], 'rb') as handle:\n",
    "        history= pickle.load(handle)\n",
    "    print(f'\\n============================================================================================================================\\n')\n",
    "    print(f\"Loading successful!\")\n",
    "    print(f'\\n============================================================================================================================\\n')\n",
    "\n",
    "else:\n",
    "    print(f'\\n============================================================================================================================\\n')\n",
    "    print(f\"Nothing to load. Training of new model... \")\n",
    "    print(f'\\n============================================================================================================================\\n')\n",
    "\n",
    "    history=model.train(nepochs, bs, lr, print_freq, h_path, pretrained, on_file=True)\n",
    "    save(model, optimizer, res_paths['weights'])\n",
    "    write(history, lc_path)\n",
    "\n",
    "    print(f'\\n============================================================================================================================\\n')\n",
    "    print(f\"Training successful!\")\n",
    "    print(f'\\n============================================================================================================================\\n')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "4a949452-ab47-4670-9245-c646cc7b3693",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": "<Figure size 640x480 with 1 Axes>",
      "image/png": "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"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "bins = 10 # binning to plot training history.\n",
    "\n",
    "if history:\n",
    "    plt.plot(np.asarray(history['ESS']).reshape(int((nepochs[0]*nlevels+nepochs[-1])/bins),bins).mean(-1))\n",
    "    plt.ylabel('ESS')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5c51a028-9e89-4544-a528-2aa80f57b118",
   "metadata": {},
   "source": [
    "-------------\n",
    "-------------\n",
    "# Analysis: Reverse Metrics\n",
    "This sections includes metrics which require samples from the model, e.g., $s\\sim q_\\theta$ where $q_\\theta$ is the autoregressive neural network.\n",
    "The resulting script `res_paths['sim']` will store:\n",
    "\n",
    "`Loss, FreeEn, ESS_rev, InternalEn, Absmag`"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "ba045204-fb9d-4fcd-bfd3-bdadcd951ccd",
   "metadata": {},
   "outputs": [],
   "source": [
    "model.eval()\n",
    "w,E,m,t = model.sample_n_OBS(100,1000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "d12ab2f7-55e3-4532-b794-7aac9e3ff4d4",
   "metadata": {},
   "outputs": [],
   "source": [
    "gamma_analysis(w, res_paths['sim']) # Stores data: Loss, FreeEn, ESS_rev\n",
    "_=gamma_analysis_OBS(E, w, res_paths['sim']) # Stores data: Internal Energy\n",
    "_=gamma_analysis_OBS(np.abs(m), w, res_paths['sim']) # Stores data: Absolute magnetization"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0975e73c-b88e-43c4-bc8c-8f03619da4ba",
   "metadata": {},
   "source": [
    "-------------\n",
    "-------------\n",
    "# Analysis: Forward metrics\n",
    "> Note to user:\n",
    "> - This only works for small lattices. As this is just a demo, configurations for larger lattices (or different $\\beta$) are not provided and users have to generate reference configurations by themselves. Once configurations are available the pipeline below can be used.\n",
    "> - Ensure to unzip the configurations before running the code below.\n",
    "\n",
    "\n",
    "This sections includes metrics which require samples from the true distribution, e.g., $s\\sim p$ where $p$ is the target Boltzmann distribution.\n",
    "Such samples can be obtained with standard methods such as the Cluster method. For the sake of this demo, configs sampled with cluster method are stored in `data/config/Ising_data_nx16_beta0.4400000000_data1000000.dat`.\n",
    "\n",
    "The script `res_paths['sim_md']` will store:\n",
    "\n",
    "`Loss_rev, FreeEn_rev, ESS_rev, Loss_fwd, FreeEn_fwd, ESS_fwd, mode_dropping_est,  InternalEn, Absmag`\n",
    "\n",
    "The script `res_paths['cluster']` will store analysis performed on samples from Cluster method, i.e., $s\\sim p$:\n",
    "\n",
    "`Loss, FreeEn, ESS_rev, InternalEn, Absmag`\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "005731df-a09d-4a20-b0f5-58b27a7aa453",
   "metadata": {},
   "outputs": [],
   "source": [
    "ndat = 1000000\n",
    "path_ising = main_path + f'config/Ising_data_nx{model.Lf}_beta0.4400000000_data{ndat}.dat'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e0e67ca9-7603-422e-ad35-9c313102d8a4",
   "metadata": {
    "pycharm": {
     "is_executing": true
    }
   },
   "outputs": [],
   "source": [
    "# Loads data for 16x16\n",
    "# N.B. This might take some time.\n",
    "data = np.genfromtxt(path_ising).reshape(-1, Lf, Lf)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bd610b06-1d9b-470c-9d5e-f4af3ad676c4",
   "metadata": {
    "pycharm": {
     "is_executing": true
    }
   },
   "outputs": [],
   "source": [
    "cluster_analysis(data, ising_energy, beta, res_paths['cluster']) # U, tau |m|, tau |m|"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "eb480693-5a2a-4188-8c17-7a6700b41fc6",
   "metadata": {
    "pycharm": {
     "is_executing": true
    }
   },
   "outputs": [],
   "source": [
    "wf = model.sample_from_MCMC(data, 1000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e62b5abf-1f37-4ddd-bc78-320de50c518e",
   "metadata": {
    "pycharm": {
     "is_executing": true
    }
   },
   "outputs": [],
   "source": [
    "gamma_analysis_modedrop(w, wf, res_paths['sim_md']) # Stores data: Loss, betaF, ESS, Floss, FbetaF, FESS, modedrop, U, absmag"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "031f75ad-397a-4db0-b08c-ff0108a0445d",
   "metadata": {},
   "source": [
    "-------------\n",
    "-------------\n",
    "# Independent Metropolis-Hastings\n",
    "\n",
    "The cells below run Neural MCMC using a Metropolis Hastings accept-reject steps to unbias the samples drawn from the model as proposed in [Nicoli et al. Phys. Rev. E (2020)](https://link.aps.org/accepted/10.1103/PhysRevE.101.023304). A cluster analysis is performed on the accepted samples and results are saved in `res_paths['mh']`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "132d59a0-c66e-47dc-9faf-bb226aee5130",
   "metadata": {
    "pycharm": {
     "is_executing": true
    }
   },
   "outputs": [],
   "source": [
    "model.eval()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2fe86663-898e-485b-9079-acb71707f73c",
   "metadata": {
    "pycharm": {
     "is_executing": true
    }
   },
   "outputs": [],
   "source": [
    "ensemble=make_mcmc_ensemble(model,100, 1000,model.device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0c75035f-d681-4f52-aabd-0ad27b6e277d",
   "metadata": {
    "pycharm": {
     "is_executing": true
    }
   },
   "outputs": [],
   "source": [
    "np.asarray(ensemble['accepted']).mean()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "73983d3d-13ee-42fc-9530-a050567953d3",
   "metadata": {
    "pycharm": {
     "is_executing": true
    }
   },
   "outputs": [],
   "source": [
    "cluster_analysis(np.asarray(ensemble['x']).reshape((-1, Lf, Lf) ), model.energy, model.beta, res_paths['mh']) #U, tau |m|, tau |m|"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "name": "multilevel",
   "language": "python",
   "display_name": "multilevel"
  },
  "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.9.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
