{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "\n",
    "os.chdir('..')\n",
    "\n",
    "import tqdm\n",
    "\n",
    "import numpy as np\n",
    "import tensorflow as tf\n",
    "\n",
    "# Get rid of the deprecation warnings\n",
    "tf.compat.v1.logging.set_verbosity(tf.compat.v1.logging.ERROR)\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib as mpl\n",
    "\n",
    "#from tensorflow.python.lib.io import tf_record\n",
    "#from tensorflow.core.util import event_pb2\n",
    "#from tensorflow.python.framework import tensor_util\n",
    "\n",
    "from models.models import DeepStatisticalSolver"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This notebook considers the AC Power Flow problem on two possible datasets.\n",
    "\n",
    "The current notebook is composed of the following parts:\n",
    "- Reload a trained model\n",
    "- Compute the important metrics (same as the ones displayed in the paper)\n",
    "- Visualize the evolution of the loss, latent variables and actual prediction\n",
    "\n",
    "By default, the notebook was made for the power grid IEEE case14, but if you want to switch to case118,\n",
    "just change the `model_path` and `data_dir` variables"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Reloading a model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Enter path to model. Three pre-trained models are available : best_linear_0, best_linear_1, best_linear_2\n",
    "model_path = \"results/best_acpf_14_0\" # \"results/best_acpf_118_0\" change for case118\n",
    "\n",
    "# Enter path to data to build architecture\n",
    "data_dir = 'datasets/acpf_14' # \"datasets/acpf_118\" change for case118\n",
    "\n",
    "# Initialize a tensorflow session\n",
    "sess = tf.Session()\n",
    "\n",
    "# Build the Deep Statistical Solver\n",
    "model = DeepStatisticalSolver(sess, \n",
    "                          model_to_restore=model_path, \n",
    "                          default_data_directory=data_dir)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load the test set (change the mode variable if you want val or train)\n",
    "mode = 'test'\n",
    "\n",
    "# Import numpy data\n",
    "A_np = np.load(os.path.join(data_dir, 'A_'+mode+'.npy'))\n",
    "B_np = np.load(os.path.join(data_dir, 'B_'+mode+'.npy'))\n",
    "U_np = np.load(os.path.join(data_dir, 'U_'+mode+'.npy'))\n",
    "coord_np = np.load(os.path.join(data_dir, 'coord_'+mode+'.npy'))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Computing metrics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# We need this short method to compute the flows based on the voltage angle and magnitude\n",
    "def compute_flows(A, B, Vm, Va):\n",
    "    \n",
    "    A_offset = A \\\n",
    "        + np.shape(B)[1] \\\n",
    "        * np.reshape(np.arange(np.shape(A)[0]), [-1, 1, 1]) \\\n",
    "        * np.array([[[1., 1., 0., 0.]]])\n",
    "    \n",
    "    i_from = A_offset[:, :, 0].astype(np.int32)\n",
    "    i_to = A_offset[:, :, 1].astype(np.int32)\n",
    "    A1 = A_offset[:, :, 2]\n",
    "    A2 = A_offset[:, :, 3]\n",
    "\n",
    "    V_i = Vm[i_from]#, 0]\n",
    "    V_j = Vm[i_to]#, 0]\n",
    "\n",
    "    theta_i = Va[i_from]#, 1]\n",
    "    theta_j = Va[i_to]#, 1]\n",
    "\n",
    "    P_ij = A1 * V_i * V_j * np.cos(theta_i - theta_j - A2)\n",
    "\n",
    "    Q_ij = A1 * V_i * V_j * np.sin(theta_i - theta_j - A2)\n",
    "    \n",
    "    return P_ij, Q_ij"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# For the whole dataset, we will compute both the individual loss of each sample, \n",
    "# and the final predictions\n",
    "costs_DSS = None\n",
    "costs_NR = None\n",
    "\n",
    "Vm_DSSs = None\n",
    "Va_DSSs = None\n",
    "Vm_NRs = None\n",
    "Va_NRs = None\n",
    "\n",
    "P_ij_DSSs = None\n",
    "Q_ij_DSSs = None\n",
    "P_ij_NRs = None\n",
    "Q_ij_NRs = None\n",
    "\n",
    "# In order to split the dataset, define the size of the batches that will be fed to the model.\n",
    "# For very large dataset, it can be useful to have a small value for BATCH_SIZE\n",
    "BATCH_SIZE = 100\n",
    "\n",
    "# Get total amount of samples and split the dataset\n",
    "n_samples_tot = A_np.shape[0]\n",
    "n_batches = n_samples_tot // BATCH_SIZE\n",
    "batched_indices = np.array_split(np.arange(n_samples_tot), n_batches) \n",
    "\n",
    "# Iterate over the batches\n",
    "for indices in tqdm.tqdm(batched_indices):\n",
    "    \n",
    "    U_DSS, cost_DSS = sess.run([model.U_final, model.cost_per_sample[str(model.correction_updates)]], \n",
    "             feed_dict = {model.A:A_np[indices], model.B:B_np[indices]}\n",
    "            )\n",
    "    \n",
    "    cost_NR = sess.run(model.cost_per_sample[str(model.correction_updates)], \n",
    "             feed_dict = {model.A:A_np[indices], model.B:B_np[indices], model.U_final:U_np[indices]}\n",
    "            )\n",
    "        \n",
    "    # Getting predicted Vm\n",
    "    Vm_DSS = U_DSS[:, :, 0]\n",
    "    Vm_DSS = np.reshape(Vm_DSS, -1)\n",
    "\n",
    "    # Getting predicted Va\n",
    "    Va_DSS = U_DSS[:, :, 1]\n",
    "    Va_DSS = Va_DSS - Va_DSS[:, 0:1]\n",
    "    Va_DSS = np.reshape(Va_DSS, -1)\n",
    "\n",
    "    # Getting actual Vm\n",
    "    Vm_NR = U_np[indices, :, 0]\n",
    "    Vm_NR = np.reshape(Vm_NR, -1)\n",
    "\n",
    "    # Getting actual Va\n",
    "    Va_NR = U_np[indices, :, 1]\n",
    "    Va_NR = Va_NR - Va_NR[:, 0:1]\n",
    "    Va_NR = np.reshape(Va_NR, -1)\n",
    "\n",
    "    # Correcting disconnected nodes\n",
    "    Va_DSS = Va_DSS * (Va_NR !=0.)\n",
    "    \n",
    "    P_ij_DSS, Q_ij_DSS = compute_flows(A_np[indices], \n",
    "                                         B_np[indices], \n",
    "                                         Vm_DSS, Va_DSS)\n",
    "    P_ij_NR, Q_ij_NR = compute_flows(A_np[indices], \n",
    "                                     B_np[indices], \n",
    "                                     Vm_NR, Va_NR)\n",
    "    P_ij_DSS = np.reshape(P_ij_DSS, -1)\n",
    "    Q_ij_DSS = np.reshape(Q_ij_DSS, -1)\n",
    "    P_ij_NR = np.reshape(P_ij_NR, -1)\n",
    "    Q_ij_NR = np.reshape(Q_ij_NR, -1)\n",
    "\n",
    "\n",
    "    if costs_DSS is None:\n",
    "        costs_DSS = cost_DSS\n",
    "        costs_NR = cost_NR\n",
    "        Vm_DSSs = Vm_DSS\n",
    "        Va_DSSs = Va_DSS\n",
    "        Vm_NRs = Vm_NR\n",
    "        Va_NRs = Va_NR\n",
    "        P_ij_DSSs = P_ij_DSS\n",
    "        Q_ij_DSSs = Q_ij_DSS\n",
    "        P_ij_NRs = P_ij_NR\n",
    "        Q_ij_NRs = Q_ij_NR\n",
    "        \n",
    "    else:\n",
    "        costs_DSS = np.concatenate([costs_DSS, cost_DSS])\n",
    "        costs_NR = np.concatenate([costs_NR, cost_NR])\n",
    "        Vm_DSSs = np.concatenate([Vm_DSSs, Vm_DSS])\n",
    "        Va_DSSs = np.concatenate([Va_DSSs, Va_DSS])\n",
    "        Vm_NRs = np.concatenate([Vm_NRs, Vm_NR])\n",
    "        Va_NRs = np.concatenate([Va_NRs, Va_NR])\n",
    "        P_ij_DSSs = np.concatenate([P_ij_DSSs, P_ij_DSS])\n",
    "        Q_ij_DSSs = np.concatenate([Q_ij_DSSs, Q_ij_DSS])\n",
    "        P_ij_NRs = np.concatenate([P_ij_NRs, P_ij_NR])\n",
    "        Q_ij_NRs = np.concatenate([Q_ij_NRs, Q_ij_NR])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print('Loss - GNS')\n",
    "print('    10th percentile = {}'.format(np.percentile(costs_DSS, 10)))\n",
    "print('    50th percentile = {}'.format(np.percentile(costs_DSS, 50)))\n",
    "print('    90th percentile = {}'.format(np.percentile(costs_DSS, 90)))\n",
    "print('Loss - NR')\n",
    "print('(Keep in mind that the maximum precision achieved by tf.float32 is around 1e-14')\n",
    "print('Thus the loss computed by tensorflow is noisy)')\n",
    "print('    10th percentile = {}'.format(np.percentile(costs_NR, 10)))\n",
    "print('    50th percentile = {}'.format(np.percentile(costs_NR, 50)))\n",
    "print('    90th percentile = {}'.format(np.percentile(costs_NR, 90)))\n",
    "print('Correlation between methods')\n",
    "print('    Correlation Vm = {}'.format(np.corrcoef(Vm_DSSs,Vm_NRs)[1,0]))\n",
    "print('    Correlation Va = {}'.format(np.corrcoef(Va_DSSs,Va_NRs)[1,0]))\n",
    "print('    Correlation Pij = {}'.format(np.corrcoef(P_ij_DSSs,P_ij_NRs)[1,0]))\n",
    "print('    Correlation Qij = {}'.format(np.corrcoef(Q_ij_DSSs,Q_ij_NRs)[1,0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "P_ij_NR.shape"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Depending on whether we are using case14 or case118, the correlations of our prediction with the output of a Newton-Raphson method should vary a bit.\n",
    "\n",
    "However, the correlation for the active and reactive power flows (i.e. Pij and Qij) should be above 99.99%"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Visualize loss, H and U"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Let's pick a random sample\n",
    "sample = np.random.randint(0,A_np.shape[0])\n",
    "\n",
    "# Predict and gather latent variables, intermediate predictions and the loss\n",
    "feed_dict = {model.A:A_np[sample:sample+1], model.B:B_np[sample:sample+1]}\n",
    "H_list, U_list, loss_list = sess.run([model.H, model.U, model.loss], feed_dict = feed_dict)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Convert those dictionaries into actual np array\n",
    "U = np.array([u[0,:,:] for u in U_list.values()])\n",
    "H = np.array([h[0,:,:] for h in H_list.values()])\n",
    "loss = np.array([l for l in loss_list.values()])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Loss visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plotting the loss\n",
    "plt.plot(np.arange(1,loss.shape[0]+1),loss)\n",
    "plt.scatter(np.arange(1,loss.shape[0]+1), loss, marker='+')\n",
    "plt.yscale('log')\n",
    "plt.ylabel('Intermediate losses')\n",
    "plt.xlabel(r'update $k$')\n",
    "plt.title('Evolution of the loss across the Deep Statistical Solver')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The loss decreases by several orders of magnitude between the first prediction and the last"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Structure of the graph"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "edges_or = A_np[sample, :, 0].astype(np.int32)\n",
    "edges_ex = A_np[sample, :, 1].astype(np.int32)\n",
    "\n",
    "# Creating mesh coordinates\n",
    "coordinates = coord_np[0]\n",
    "coordinates_or_x = coordinates[edges_or, 0]\n",
    "coordinates_or_y = coordinates[edges_or, 1]\n",
    "coordinates_ex_x = coordinates[edges_ex, 0]\n",
    "coordinates_ex_y = coordinates[edges_ex, 1]\n",
    "\n",
    "coordinates_edge_x = np.c_[coordinates_or_x, coordinates_ex_x]\n",
    "coordinates_edge_y = np.c_[coordinates_or_y, coordinates_ex_y]\n",
    "\n",
    "plt.figure(figsize=[5,5])\n",
    "plt.tick_params(\n",
    "    axis='x',          # changes apply to the x-axis\n",
    "    which='both',      # both major and minor ticks are affected\n",
    "    bottom=False,      # ticks along the bottom edge are off\n",
    "    top=False,         # ticks along the top edge are off\n",
    "    labelbottom=False) # labels along the bottom edge are off\n",
    "plt.tick_params(\n",
    "    axis='y',          # changes apply to the x-axis\n",
    "    which='both',      # both major and minor ticks are affected\n",
    "    bottom=False,      # ticks along the bottom edge are off\n",
    "    top=False,         # ticks along the top edge are off\n",
    "    left=False,\n",
    "    right=False,\n",
    "    labelleft=False) # labels along the bottom edge are off\n",
    "\n",
    "plt.plot(coordinates_edge_x.T, coordinates_edge_y.T, 'k', zorder=1)\n",
    "plt.title(r'Mesh $\\tilde{\\mathbb{G}}$')\n",
    "plt.show()\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This is the undirected and unweighted graph that bears the structure of the problem instance.\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Intermediate predictions of Voltage module"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "max_x = np.max(U[:,:, 0])\n",
    "min_x = np.min(U[:,:, 0])\n",
    "\n",
    "for frame in range(U.shape[0]):\n",
    "    plt.figure(figsize=[3,2.5], dpi=300)\n",
    "    \n",
    "    plt.tick_params(\n",
    "        axis='x',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        labelbottom=False) # labels along the bottom edge are off\n",
    "    plt.tick_params(\n",
    "        axis='y',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        left=False,\n",
    "        right=False,\n",
    "        labelleft=False) # labels along the bottom edge are off\n",
    "    plt.plot(coordinates_edge_x.T, coordinates_edge_y.T, 'k', zorder=1)\n",
    "    plt.scatter(coordinates[:,0], coordinates[:,1], c=U[frame,:, 0], zorder=2)\n",
    "    plt.clim(min_x, max_x)\n",
    "    plt.colorbar()\n",
    "    plt.title(r'Intermediate prediction $\\widehat{\\mathbf{U}}^{'+str(frame+1)+'}$')\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Intermediate predictions of Voltage angle"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "max_x = np.max(U[:,:,1]-U[:,0:1, 1])\n",
    "min_x = np.min(U[:,:,1]-U[:,0:1, 1])\n",
    "\n",
    "for frame in range(U.shape[0]):\n",
    "    plt.figure(figsize=[3,2.5], dpi=300)\n",
    "    \n",
    "    plt.tick_params(\n",
    "        axis='x',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        labelbottom=False) # labels along the bottom edge are off\n",
    "    plt.tick_params(\n",
    "        axis='y',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        left=False,\n",
    "        right=False,\n",
    "        labelleft=False) # labels along the bottom edge are off\n",
    "    plt.plot(coordinates_edge_x.T, coordinates_edge_y.T, 'k', zorder=1)\n",
    "    plt.scatter(coordinates[:,0], coordinates[:,1], c=U[frame,:, 1]-U[frame,0, 1], zorder=2)\n",
    "    plt.clim(min_x, max_x)\n",
    "    plt.colorbar()\n",
    "    plt.title(r'Intermediate prediction $\\widehat{\\mathbf{U}}^{'+str(frame)+'}$')\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Latent variables evolution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Select which of the d components of H you want to visualize\n",
    "component_of_H = 2\n",
    "\n",
    "max_h = np.max(H[:,:,component_of_H])\n",
    "min_h = np.min(H[:,:,component_of_H])\n",
    "\n",
    "for frame in range(H.shape[0]):\n",
    "    plt.figure(figsize=[3,2.5], dpi=300)\n",
    "    \n",
    "    plt.tick_params(\n",
    "        axis='x',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        labelbottom=False) # labels along the bottom edge are off\n",
    "    plt.tick_params(\n",
    "        axis='y',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        left=False,\n",
    "        right=False,\n",
    "        labelleft=False) # labels along the bottom edge are off\n",
    "    \n",
    "    plt.plot(coordinates_edge_x.T, coordinates_edge_y.T, 'k', zorder=1)\n",
    "    plt.scatter(coordinates[:,0], coordinates[:,1], c=H[frame,:,component_of_H], zorder=2)\n",
    "    plt.clim(min_h, max_h)\n",
    "    plt.colorbar()\n",
    "    plt.title(r'$\\mathbf{H}^{'+str(frame+1)+'} - component {'+str(component_of_H)+'}$')\n",
    "    plt.show()"
   ]
  },
  {
   "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.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
