{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "import heapq\n",
    "from collections import defaultdict\n",
    "import math\n",
    "import networkx as nx\n",
    "import matplotlib.pyplot as plt\n",
    "import itertools\n",
    "import timeit\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "class ExistsDirectedBackdoorPath(Exception):\n",
    "    \"\"\"Exception raised when a directed backdoor path is found.\"\"\"\n",
    "    def __init__(self, message=\"A directed backdoor path exists.\"):\n",
    "        self.message = message\n",
    "        super().__init__(self.message)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Implementation of Algorithm 1\n",
    "\n",
    "All the implementations use $\\gamma$ instead of $t - \\gamma$ because $t$ is a silent variable."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Computation of $t_{NC}$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_AncY(G, Interventions):\n",
    "    \"\"\"\n",
    "    Compute \\left(\\max \\left\\{ t_1 \\mid  \\exists\\Gf\\text{ s.t } \\Gf \\setminus \\Xf \\models S_{t_1} \\rightsquigarrow Y_{t}  \\right\\}\\right)_{S \\in \\mathcal{V}^s}.\n",
    "    \n",
    "    Parameters:\n",
    "    - G: NetworkX DiGraph representing the SCG.\n",
    "    - Interventions: list of tuples [(V, gamma)], representing the interventions.\n",
    "\n",
    "    Returns:\n",
    "    - AncY: dict with nodes as keys and their t - t^NC_{V_{t- gamma_v}}(S) values as values.\n",
    "    \"\"\"\n",
    "    \n",
    "    if not G.has_node('Y'):\n",
    "        raise ValueError(f\"The node {Y} is not in the graph G.\")\n",
    "    \n",
    "    # Priority queue and initialization\n",
    "    Q = []\n",
    "    heapq.heappush(Q, (0, 'Y'))  # Min-heap by using gamma\n",
    "    AncY = {node: math.inf for node in G.nodes}  # Initialize the value for Y  \n",
    "    seen = {node: False for node in G.nodes}  # Track if a node has been processed\n",
    "    \n",
    "    AncY['Y'] = 0  # Initialize the value for Y\n",
    "    seen['Y'] = True  # Mark Y as seen\n",
    "\n",
    "    # Main processing loop\n",
    "    while Q:\n",
    "        gamma_s, S = heapq.heappop(Q)  # Extract node with the smallest gamma\n",
    "        \n",
    "        # Process all unseen parents of S\n",
    "        for P in G.predecessors(S):  # Access parents using NetworkX\n",
    "            if not seen[P]:\n",
    "                # Compute AncY for P\n",
    "                gamma_AncY_P = calculate_min_gamma_not_intervention_above_P_t_gammaP(P, gamma_s, Interventions)\n",
    "                \n",
    "                if gamma_AncY_P != math.inf:  \n",
    "                    heapq.heappush(Q, (gamma_AncY_P, P))  # Add parent to the queue\n",
    "                    AncY[P] = gamma_AncY_P  # Update t^NC value for P\n",
    "                \n",
    "                seen[P] = True  # Mark parent as seen\n",
    "\n",
    "    return AncY\n",
    "\n",
    "def calculate_max_gamma_not_intervention_under_P_t_gammaP(P, gammaP, Interventions):\n",
    "    if (P, gammaP) in Interventions:\n",
    "        # Extract all gamma values associated with P in Interventions\n",
    "        Interventions_gammas = {g for node, g in Interventions if node == P}\n",
    "        \n",
    "        # Find the largest gamma' < gamma that is not in Interventions\n",
    "        for gamma_prime in range(gammaP - 1, -1, -1):  # Iterate from gamma-1 to 0\n",
    "            if gamma_prime not in Interventions_gammas:\n",
    "                return gamma_prime\n",
    "        return(-math.inf) \n",
    "    else: \n",
    "        return gammaP\n",
    "    \n",
    "def calculate_min_gamma_not_intervention_above_P_t_gammaP(P, gammaP, Interventions):\n",
    "    if (P, gammaP) in Interventions:\n",
    "        # Extract all gamma values associated with P in Interventions\n",
    "        Interventions_gammas = {g for node, g in Interventions if node == P}\n",
    "        \n",
    "        # Find the largest gamma' < gamma that is not in Interventions\n",
    "        gamma_prime = gammaP + 1\n",
    "        while 1:\n",
    "            if gamma_prime not in Interventions_gammas:\n",
    "                return gamma_prime\n",
    "            gamma_prime += 1\n",
    "    else: \n",
    "        return gammaP\n",
    "    \n",
    "    \n",
    "\n",
    "\n",
    "def calculate_d(gamma_C, interventions_in_Pa_C):\n",
    "    \"\"\"\n",
    "    Calculate d(C) for a single node C in the graph G.\n",
    "\n",
    "    Parameters:\n",
    "    - G: NetworkX DiGraph representing the graph.\n",
    "    - Interventions: list of tuples [(X^i, gamma_i)], representing the interventions.\n",
    "    - gamma_C: t_C = t - \\gamma_C.\n",
    "\n",
    "    Returns:\n",
    "    - d_C: boolean value indicating if d(C) satisfies the conditions.\n",
    "    \"\"\"\n",
    "    count_equal = 0\n",
    "\n",
    "    for _, gamma_i in interventions_in_Pa_C:\n",
    "        if gamma_i > gamma_C:\n",
    "            return True\n",
    "        if gamma_i >= gamma_C:\n",
    "            count_equal += 1\n",
    "        if count_equal >= 2:\n",
    "            return True\n",
    "    return False\n",
    "\n",
    "\n",
    "def compute_t_NC(G, Interventions):\n",
    "    \"\"\"\n",
    "    Compute (t_NC(S)) for all S in V^s.\n",
    "\n",
    "    Parameters:\n",
    "    - G: networkx.DiGraph representing the SCG.\n",
    "    - Interventions: list of tuples [(V, gamma)], representing the interventions.\n",
    "\n",
    "    Returns:\n",
    "    - t_NC: dict with nodes as keys and their t - t_NC values as values.\n",
    "    \"\"\"\n",
    "    # Sort Interventions\n",
    "    Interventions.sort(key=lambda x: x[1], reverse=True)\n",
    "    \n",
    "    AncY = compute_AncY(G, Interventions)\n",
    "\n",
    "    Cs = set()\n",
    "    for I,_ in Interventions:\n",
    "        Cs.update(G.successors(I))\n",
    "    gamma_C = {}\n",
    "    d = {}\n",
    "    for C in Cs:\n",
    "        #computes t_C[C]\n",
    "        interventions_in_Pa_C = [(X, gamma) for X, gamma in Interventions if X in G.predecessors(C)]\n",
    "        gamma_max = max((gamma for X, gamma in interventions_in_Pa_C), default=-math.inf)\n",
    "        gamma_C_temp = calculate_max_gamma_not_intervention_under_P_t_gammaP(C, gamma_max, Interventions)\n",
    "        if gamma_C_temp >= AncY[C]:\n",
    "            gamma_C[C] = gamma_C_temp\n",
    "        else:\n",
    "            gamma_C[C] = -math.inf\n",
    "            \n",
    "        #computes d[C]\n",
    "        d[C] = calculate_d(gamma_C[C], interventions_in_Pa_C)\n",
    "\n",
    "    #sorts L\n",
    "    L = [(C, gamma_C[C]) for C in Cs if gamma_C[C] > -math.inf]\n",
    "    L.sort(key=lambda x: (-x[1], not d[x[0]]))\n",
    "    \n",
    "    # computes t_NC\n",
    "    # Initialize t_NC and 'seen' markers\n",
    "    gamma_NC = {node: -math.inf for node in G.nodes}\n",
    "    seen = {node: False for node in G.nodes}\n",
    "    X = set(X for X, _ in Interventions)    \n",
    "    # Iterate over sorted Interventions\n",
    "    for C, gamma_C in L:\n",
    "        if d[C]:\n",
    "            for D in find_unseen_descendants(G, C, seen):\n",
    "                gamma_NC[D] = calculate_max_gamma_not_intervention_under_P_t_gammaP(D, gamma_C, Interventions)\n",
    "                seen[D] = True\n",
    "        else:\n",
    "            for D in find_unseen_descendants_notin_X(G, C, seen, X):\n",
    "                gamma_NC[D] = calculate_max_gamma_not_intervention_under_P_t_gammaP(D, gamma_C, Interventions)\n",
    "                seen[D] = True\n",
    "            for D in find_unseen_descendants(G, C, seen):\n",
    "                gamma_NC[D] = calculate_max_gamma_not_intervention_under_P_t_gammaP(D, gamma_C, Interventions)\n",
    "                seen[D] = True\n",
    "    return gamma_NC\n",
    "\n",
    "def find_unseen_descendants(G, start_node, seen):\n",
    "    \"\"\"\n",
    "    Find  descendants of start_node in a graph G.\n",
    "\n",
    "    Parameters:\n",
    "    - G: networkx.DiGraph representing the SCG.\n",
    "    - start_node: the node to find unseen strict descendants for.\n",
    "\n",
    "    Returns:\n",
    "    - descendants: set of unseen strict descendants (including start_node if part of a cycle).\n",
    "    \"\"\"\n",
    "    visited = []\n",
    "    stack = [start_node]\n",
    "    while stack:\n",
    "        node = stack.pop()\n",
    "        if not seen[node]:\n",
    "            seen[node] = True\n",
    "            visited.append(node)\n",
    "            stack.extend(G.successors(node))  # Add neighbors to the stack\n",
    "    return visited\n",
    "\n",
    "def find_unseen_descendants_notin_X(G, start_node, seen, X):\n",
    "    \"\"\"\n",
    "    Find  descendants of start_node in a graph G \\ X.\n",
    "\n",
    "    Parameters:\n",
    "    - G: networkx.DiGraph representing the SCG.\n",
    "    - start_node: the node to find unseen strict descendants for.\n",
    "\n",
    "    Returns:\n",
    "    - descendants: set of unseen strict descendants (including start_node if part of a cycle).\n",
    "    \"\"\"\n",
    "    visited = []\n",
    "    stack = [start_node]\n",
    "    while stack:\n",
    "        node = stack.pop()\n",
    "        if not seen[node] and node not in X:\n",
    "            seen[node] = True\n",
    "            visited.append(node)\n",
    "            stack.extend(G.successors(node))  # Add neighbors to the stack\n",
    "    return visited\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAgMAAAGFCAYAAABg2vAPAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8o6BhiAAAACXBIWXMAAA9hAAAPYQGoP6dpAABaJUlEQVR4nO3dd1hcd5on+u+pSM5RJAEiJ2UJBZRztCRLtmzLsiy5t2fHs3Nnd57ee3dne3fu3Ul38s5sP9eWQ/eM2+5RlgAllIMVrAwFKAOCIueCggrn/oEpC5EKKDgVvp/n6ccyVB1e3KVT3/qd97w/QRRFEUREROSyZFIXQERERNJiGCAiInJxDANEREQujmGAiIjIxTEMEBERuTiGASIiIhfHMEBEROTiGAaIiIhcHMMAERGRi2MYICIicnEMA0RERC6OYYCIiMjFMQwQERG5OIYBIiIiF8cwQERE5OIYBoiIiFwcwwAREZGLYxggIiJycQwDRERELo5hgIiIyMUxDBAREbk4hgEiIiIXxzBARETk4hgGiIiIXBzDABERkYtjGCAiInJxDANEREQujmGAiIjIxTEMEBERuTiGARo3oihKXQIREVlBIXUB5DyqO4x42KDHq3YD6rtMMIuATACC1HJEeimRGeiGMA++5IiI7I0g8uMbjVFTlwn5ZW2o0BkhA2Ae4DG9X4/yVGBtjDf81fKJLZKIiAbFMEBjUtSoR355O8wiYM0LSUDPasG6aG+kBqjHuzwiIrICwwCNWlGjHifK2kf9/A0xXkgLcLNhRURENBpsIKRRadSbkF8++iAAAPnl7WjqMtmoIiIiGi2GARqVk+VtMI9xTcksAvllbbYpiIiIRo2t3TRi1R1GVOiMg37/L9dNh0KlglLtBmN3NyYlZ2DLn/wtVO6efR4nAqjQGVHdYeRdBkREEuLKAI3Ywwb9sC+cd//ic/zBdxfxhwevQt/eijvHvxvwcbIfj0dERNJhGKARe9VuGPD2wYGYDN0w6Dvh7uM34PfNAF7pDLYqjYiIRoFrszRi9VY0/X37n/dBqXZDU1UFIlKykLFi0+DH07OJkIhISlwZoBERRdGqxsHeywT/9Xwp/CdF4dQ//umgjzWLHF1MRCQlhgEaEUEQIBOsf7xcoUDasvV4fP38oI+RCT3HJSIiaTAM0IgFjXCU8PPbVxEcM2Xw47lxNDERkZTYM0AjFumlRL3eNGQTYW/PgNlkgl94JDb/X3894ONkACI9leNSJxERWYfjiGnEqjuM+Lq02WbH253kxzkDREQS4mUCGrEwDwWiPBUY61V+AT27GDIIEBFJi2GARmVtjPeIGgkHIhN6jkNERNJiGKBR8VfLsS56bG/k66K94T/CZkQiIrI9rs/SqKUGqCFCRH55e8+sACueI6BnRWBdtDdSA9QAgOvXr6OzsxOZmZkICgribYZERBOMDYQ0Zk1dJuSXtaFCZ4QMGPAug96vR3kpsPaNFYG///u/R0tLCwBArVYjJiYGMTExiI6ORnh4OORyrh4QEY0nhgGymeoOIx426PFKZ+i59VDsWQUIcpMj0lOJzEC3AZsFS0pK8Lvf/c7y770rA6IoQi6XIyIiwhIOoqKioFarJ+x3IiJyBQwDNG5EUbRqyV8URfzDP/yDZXVgIDKZDGZzz5pDSEgIli5diqSkJJvVSkTkythASOPG2mv/giBg3rx5Qz6mNwgAQG1tLVpbW8dUGxER/YRhgMbF1atXUVBQYPXjs7KyoFQOP4lQEASkp6dj5syZYymPiIhewzBANtfZ2YkLFy5Y7hKwhlqtxowZM4ZcTRAEAWFhYdi0aRPvOCAisiGGAbK5q1evwmw2QxRF/PDDD1Y/b/bs2UNuZezp6YmdO3dCoeAdsUREtsQwQDbV2tqKmzdvWv79+++/h9FotOq5/v7+SE5OHvRTf0BAgFWXEoiIaGQYBsimLl682KfZr7OzE/fv37f6+XPnzh1wdSAnJwc1NTX48ssv0dzcbINKiYioF8MA2UxdXR3u3bvX782897KBNaKjoxEaGtpndWDp0qVYsmQJPv74YxgMBnz++ecoLy+3ae1ERK6MYYBspqCgYMAl/paWFpSUlFh1DEEQkJ2dbZlRkJGRgQULFgAAgoODsXfvXgQHB+PXv/71iFYciIhocAwDZBPl5eV4/PjxgEv8giDgypUrQzYHvi4tLQ2enp4IDw/Hxo0b+wQMDw8PfPDBB8jKysKxY8dw9uxZq1cdiIhoYJxASGMmiiK+/PJLVFZWDvmGv2vXLsTGxlp1zPb2dri5uQ1654Aoirh58ybOnDmDhIQEbNmyhWOKiYhGiSsDNGaPHz/Gq1evhgwCgiDg6tWrVh/Ty8tryFsIBUHA3Llz8e677+Lly5dsLCQiGgOGARoTs9mMs2fPDjsESBRFPH/+HDU1NTb9+QkJCfj444/R3d3NxkIiolFiGKAxKSoqQkNDg9X9ACNZHbBWSEgI9u3bh6CgIPzmN7/BgwcPbP4ziIicGUe50Zj4+/sjKSkJBoMBBoMBRqMRBoMB9fX1UKvVkMlkMJlMlv81NjaOSx0eHh7YtWsXcnNzcfToUdTV1WHZsmUcW0xEZAU2EJLNGQwG/Nmf/RneeustZGZmWr7e+1IbzzdoURRx48YNnDlzBklJSdiyZQtUKtW4/TwiImfAywRkc73jh99sABQEYdw/qffOKXj33Xfx4sULNhYSEVmBYYBsbrAwMJESExPx8ccfo6urC/v370dFRYVktRAR2TuGAbI5ewgDQE9j4d69exEYGIhf//rXePjwoaT1EBHZK4YBsjl7CQNAz7bHH3zwATIyMnDkyBGcO3fO6jsfiIhchfRna3I69hQGgJ46Nm7ciODgYJw9exb19fV466232FhIRPQjrgyQzZlMJgD2EwaAnsbCefPm4d1338Xz58/x5ZdfoqWlReqyiIjsAsMA2Zy9rQy8LjExEXv27IFer8fnn3+OV69eSV0SEZHkGAbI5uw5DABAaGgo9u3bh4CAAHz99dd49OiR1CUREUmKYYBsrjcMyOVyiSsZnKenJ3bt2oX09HQcPnyYjYVE5NLs86MbOTR7XxnopVAosGnTJgQHB6OgoICNhUTksrgyQDbnKGEA6GksnD9/Pt555x08e/YMX331FRsLicjlMAyQzRmNRgiCAJnMcV5eSUlJ+Pjjj9HZ2Yn9+/ezsZCIXIrjnK3JYRiNRigUCofbMTA0NBR79+6Fv78/GwuJyKUwDJDN9YYBR+Tl5YVdu3YhLS0Nhw8fxvnz59lYSEROzzHP2GTXjEajXd9JMByFQoHNmzcjODgY586dQ319PTZv3szGQiJyWlwZIJtz5JWBXoIgYMGCBdixYweePn2Kr7/+Gq2trVKXRUQ0LhgGyOacIQz0Sk5Oxp49e6DT6fD555+jsrJS6pKIiGyOYYBszmQyOU0YAICwsDDs27cPfn5++Prrr1FYWCh1SURENsUwQDbnTCsDvby8vPDhhx8iJSUFhw4dwoULF9hYSEROw7nO2GQXnDEMAD2NhW+99RaCg4Nx/vx5S2OhUqmUujQiojHhygDZnLOGAaCnsXDhwoXYvn07njx5gq+++oqNhUTk8BgGyOacOQz0SklJwUcffWRpLKyqqpK6JCKiUWMYIJtzhTAAAOHh4di3bx98fX3x1VdfoaioSOqSiIhGhWGAbM7Z7iYYSm9jYXJyMg4ePIiLFy+ysZCIHI5rnLFpQjn6BMKRUiqV2LJlC4KDg3HhwgXU19dj06ZNbCwkIofBlQGyOVe5TPA6QRCQk5ODt99+G48fP8bXX3+NtrY2qcsiIrIKwwDZnCuGgV6pqan46KOP0NbWxsZCInIYDANkc64cBoCfGgt9fHzw1VdfQaPRSF0SEdGQGAbI5lw9DACAt7e3pbHwwIEDuHTpEhsLichuufYZm8YFw0CP3sbCoKAgXLx4EfX19di4cSMbC4nI7vCMTTbHMPATQRCwaNEiBAcH48iRI2hsbMQ777wDb29vqUsjIrLgZQKyKbPZDFEUGQbe8GZjoVarlbokIiILhgGyKaPRCAAMAwOYNGkS9u3bB29vb3z55ZdsLCQiu8EwQDbVGwZcaejQSHh7e2P37t1ISkrCgQMHcPnyZTYWEpHk+PGNbIorA8NTKpXYunUrgoKCLBMLN2zYwMZCIpIMz9hkUwwD1hEEAYsXL0ZwcDCOHj1qaSz08vKSujQickG8TEA2xTAwMmlpadi9ezdaWlrYWEhEkmEYIJtiGBi5iIgI7Nu3D56envjqq69QXFwsdUlE5GIYBsimGAZGx8fHBx999BESEhLwb//2b7hy5QobC4lowjAMkE0xDIyeUqnEtm3bsGjRIpw/fx5Hjhyx/PckIhpPPGOTTTEMjE1vY2FQUBCOHTuGpqYm7Nixg42FRDSuuDJANsUwYBvp6enYvXs3mpub8fnnn6O6ulrqkojIiTEMkE2ZTCYADAO28Hpj4ZdffomSkhKpSyIiJ8UwQDbFCYS25ePjg927dyMhIQG/+93vcPXqVTYWEpHNMQyQTRmNRshkMshkfGnZikqlwrZt25CTk4Nz587h6NGjbCwkIpviWi7ZFLcvHh+CIGDJkiX9Ggs9PT2lLo2InAA/vpFNMQyMr4yMDOzevRtNTU34/PPPUVNTI3VJROQEGAbIphgGxl9kZCT27t0Ld3d3fPHFFygtLZW6JCJycAwDZFMMAxPD19cXH330EaZMmYLvvvsO165dY2MhEY0awwDZFMPAxFGpVHj77bexcOFCFBQU4NixY2wsJKJR4VmbbIphYGIJgoClS5ciKCgIx48fR2NjIxsLiWjEuDJANmUymThjQAKZmZnYvXs3Ghsb2VhIRCPGMEA2xZUB6URGRmLfvn1wc3PDl19+ycZCIrIawwDZFMOAtHx9fbFnzx7ExcXhu+++w/Xr19lYSETDYhggm2IYkJ5KpcL27duxYMECnD17FsePH2djIRENiWdtsimj0QgPDw+py3B5giBg2bJlCAoKwokTJ9DY2Ijt27ezsZCIBsSVAbIprgzYl6ysLHz44YdoaGjA/v37UVtbK3VJRGSHGAbIpoxGI+8msDNRUVHYu3cv1Go1vvjiCzx+/FjqkojIzjAMkE1xZcA++fn5Yc+ePYiNjcW3336L77//no2FRGTBMEA2xTBgv1QqFXbs2IH58+fjzJkzOH78OEwm05iP29HRgb/6q7/CoUOHUF9fb4NKiWii8axNNmUymRgG7JggCFi+fDmCg4Nx4sQJNDU1Yfv27WNq+uzs7ERnZycKCwtRWFiI9PR0LFq0CEFBQTasnIjGE8/aZFNcGXAMWVlZ8Pf3x+9+9zt8/vnn2LlzJ4KDg21ybI1Gg8LCQsTGxmLZsmVwc3ODTqdDYGAg72YgslM8a5NNMQw4jujoaOzbtw/ffvstvvjiC2zduhUJCQljPq7ZbAYAvHjxAgcOHIDJZEJ7ezuAnt6FtLQ0zJ07F15eXmP+WURSEUURgiBIXYbN8KxNNiOKIsOAg+ltLDx8+DC+/fZbrFixAnPnzh3zSU4ul2Pq1KlYtGgRFAoF2traUFtbi/Lycty+fRs3b97E6tWrMWPGDBv9JkTjq7rDiIcNerxqN6C+ywSzCMgEIEgtR6SXEpmBbgjzcNxzn+NWTnantxmNYcCxqNVq7NixA+fOncOZM2dQV1eHdevWWX2LaGdnp+XPCoUC2dnZyM7Ohru7u+Xr7u7uCAkJQXp6OpYuXYqCggLk5uaiqakJy5cvt/nvRGQrTV0m5Je1oUJnhAyA+bXvmUWgVm9Cvd6Eu/V6RHkqsDbGG/5qx7u9mmdtshmGAcclk8mwYsUKS2Nh78RCaxoLr1+/DkEQMGvWLCxevLhPCBiIm5sb1q9fj4CAAJw9exbh4eFIS0uz1a9CZDNFjXrkl7fD/ONduOZBHtf79Vc6I/YXN2FdtDdSA9QTUaLN8NZCspne+fcMA45r6tSp+PDDD1FXV4f9+/ejrq5uyMc/efIExcXF2Lx5M9asWTNsEHhddnY20tPTcezYMTQ1NY21dCKbKmrU40RZO0wiYO1EDhGASQSOl7WhqFE/nuXZHMMA2QzDgHOIjo7G3r17oVQq8cUXX+Dp06eDPvbhw4cICwtDRkbGiH+OIAjYsGEDZDIZHjx4MJaSiSy6urqg0+nGdIxGvQn55e1jOkZ+eTuausY+x2OiMAyQzTAMOA9/f3/s2bMH0dHR+O1vf4sbN270m1goiiKeP3+OKVOmjLrhUKVSISkpCRqNxhZlE+HUqVP4m7/5Gxw8eBCVlZWjOsbJ8jbLpYE3mYxGFPx//y/+dks2/v7thfjHdxbj8P/9R+hsa+nzOLMI5Je1jernS4FnbbIZhgHnolar8c4776CgoACnT59GXV0d1q5da2ksrK2tRUdHB+Li4sb0c1JTU/Hw4UPU1dXZbNYBuS6z2QxRFFFcXIyioiJMmjQJ2dnZSElJsaoptrrDiArd4Ft+H/rT/4DOlmb8/OuTcPfxgyiKKCw4js6WJrh7+1oeJwKo0BlR3WF0iLsM7L9Cchi9YYAbFTkPmUyGlStXIjg4GLm5uWhsbMTbb78NDw8PVFVVAejZCGks4uPjIZPJ8OLFC4YBGjOVSgWZTGaZd6HVanHo0CF4eXlhzpw5mDFjxpC9LQ8b9P3uGuhVX/4chQUn8Iu8e3D38QPQc7krY8WmAY8l+/F4YR72P1ODlwnIZrgy4LymTZuGXbt2oaamBvv370d9fT1kMutPH0NtiqRQKKBQKCwnb6KxUKv7dvH3vvba29tx/vx5/M3f/A1yc3MHbY591W4Y9K6BqpKHCIyKg6d/oFW1mAG80hmsLV1SPGuTzTAMOLeYmBjLxML9+/djzpw5AACDwdDv/3NnH9BC9kulUg36PVEUYTKZcPfuXdy5cweTJk3CrFmzMHXqVMtj6m3c9Fevd4wmQv5tJJthGHB+/v7++Pjjj3Ho0CFcvnwZANDd3W1ZdnWVAS00cYxGI7q7u9HV1WX552B/7u7uRlVV1bCrTL2rBVVVVTh27BgyMjIgl8shiuKgjYMAMCk5Ew0Vz6FrboSnX4BV9ZtFxxhdzLM22QzDgGvobSw8fvw4Hjx4gMrKSvj6+o56QMviIBm6u7u5V4GTEEURBoNh0Dfrwd7MB3v8cNtsK5VKqNVqqFQqqNVqdHd3W11reHg4Vq5caelzEgQBMgGDBoKg6DikLV2Pw3/6h9j2P/4X3L19IYoiis7nYlJSBgIiJ/d7jkyA3QcBgGGAbIhhwHXIZDJs3rwZ1dXVKCoqghgWhxNlI7svu3dAy7laExRRSTbZJIlGx2w2D/mGPNjXBvveUD0igiBY3rhffxNXqVTw9PTs8+9vPubNx/c2C75Oo9HgwIEDQ/6+fn5+WL16NRITE/u9UQep5agdYml/2y//Aef3/y3+965VkMkVEM1mxE7PRvzsnAEfH+TmGCtfPGuTzfQmeN5N4DoyMjJw4eZdvBzjgBZ51lJ0QAHHGuAqLZPJZPXS+XBv4AbD0E1ucrl8wDdkd3d3+Pr6DvpmPdCbuUKhGNdPym82EPYSBAFKpRJLlizBrFmzBj1PRXopUa83DbqyJVcqseLnv8CKn/9i2FpkACI9lVZWLi2GAbIZo9EIuVzuEEtiZBsZGRm41OENk1kERvv/uyBAEHoGtLyX6GfT+uxJ766e1rxxD7d03tXVZdXy+UBvyN7e3ggMDBzRG7gjBfw3Gwh7z0ezZs3CokWLht1vIzPQDXfrbTNK2Pzj8RwBwwDZDLcvdj0dCg/IgiYN+v0uXTv+bGUaMlduxtZf/sOgjxMh2OWAlrEun7/5mKGWzwEM+ubcu3xuzdJ57z9HcuunM3lzZSA+Ph6rVq1CUFCQVc8P81AgylOBVzqj1XsSDEQAEOmpsKvX81Aco0pyCAwDrmeoAS0A8PDMEUSkZKHwfB7W//H/hHqI4Su2GtAykcvnMplswDdnNzc3+Pj4WP3JW6VSQalUclXNBjw9PSEIAgIDA7F69WrEx8eP+BhrY7yxv7gJpjGkAZnQcxxHwTM32QzDgOsZakALANw++lss3fdHuHXoN3h45ihmbX5/0Mf2DmhpbW1FY2PjiJfPre0+VygUA74he3l5ISAgwKpP3r1/5uvd/nh6euLTTz+Fr6/vqFdH/NVyrIv2xvEx7C2wLtqxbpvlK5lshmHA9Qw1oKXmeSlaaiqRmL0UZpMJl776xyHDANAzoGX//q/Q1tb3JDzYG7KHh4fVS+euvnzuSvz9/cd8jNQANUSIlttlrVkkENCzIrAu2hupAY7VDsszN9kMw4BrGW5Ayw9Hv8H0ddshk8uRNH85jv7P/4ja548REpc46HPMIrBv3z50d3db3sS5fE4TzWw2QxAEpAW4YZKnctBBWr16vx7ppcBaB1sR6MUzN9kMw4BrGWpAi8lgwL28A5ArFLh/6jAAoFvfiR+OfYO1/8f/GPSYMgHw9nac66zkXLq6unD16lVcu3YNmZmZ2Lx5M/zVcryX6PfTiG2doefWw94R225yRHo6/ohtx62c7I7JZGIYcDGDDWgpvnwKAREx+L3fnLJ8rfb5Y3z+yWas+v3/Crly4HuvHWVACzmXrq4u3Lp1C9euXUNXVxcA9Lv7IMxD0ae51RFGDI8Ez9xkM1wZcH5NTU09EwdFEQsXLhx0QMvto99g6tqtfb4WEpcIn5AwFF8+jfRl6/sd25EGtJBz6O7uxq1bt3D16tV+t37GxsYO+VxnCgIAwwDZEMOAc2ptbUVRUREKCwtRVVUFhUKBmTNnAhh8QMtH/+u7AY/16W/PD/pzHGlACzm27u5u3L59G1evXoVeP/CAoZCQkAmuSlo8c5PNGI1Gy+515Nja29uh0WhQVFSE8vJyyOVyJCQkIDs7G4mJiZYpb646oIUckzUhAAB8fX2hHORSlrPi3zyymd5xxOSYOjo6UFxcjKKiIrx8+RKCICA+Ph6bN29GUlIS3NwG/tTuigNayPHcv38fZ86cQWdn55CPEwQB4eHhE1SV/WAYIJvhZQLHo9frUVJSgqKiIjx//hyiKGLy5MlYv349kpOTh53jDrjmgBZyPHfv3h02CAA9YSA0NHQCKrIvPHOTzTAMOIbu7m6UlpaiqKgIT58+hclkQnR0NFavXo2UlBR4eY18HLCrDWghx/P+++8jNzcXjx49GvJxZrOZYYBoLBgG7JfBYMCTJ09QVFSEx48fw2g0IiIiAsuWLUNaWhp8fHzG/DNcaUALOR6VSoW33noLsbGxyMvLg9lsHnTjKIYBojFgGLAvJpMJz549Q2FhIUpLS9Hd3Y2wsDAsWrQIaWlpNhnZ+iZXGdBCjkkQBEybNg2TJk3Cl19+ie7u7n6PUSgU4/J3w97xbyPZDIcOSc9sNuPFixcoLCxESUkJ9Ho9goODMW/ePKSnpyMwMHBC6lB0NKP7wQ3sWd8zT8DZBrSQY6usrER3dzeio6NRXl7e53vBwcEu+VrlmZtshncTSMNsNqO8vByFhYUoLi5GR0cHAgICMGvWLKSnp0tyv3RRURE0Gg3WrVsHQRBc8uRK9qmmpgYnT57E9OnTsWHDBjx48AC5ubmW3S5d8U4CgGGAbIiXCSaOKIp49eoVCgsLodFo0N7eDl9fX0ydOhXp6ekICwuT9A1Yq9UiPDycIYDsSnd3Nw4ePIiAgACsXr0aAJCVlYWIiAj87ne/Q319vcsNG+rFMzfZhCiKvEwwzkRRhFartQSAlpYWeHl5IS0tDenp6YiIiLCbN1+tVouMjAypyyDqIz8/Hy0tLfjkk0/6DBUKCgrCJ598gnv37iEzM1PCCqXDMzfZhNFoBACGARsTRRG1tbWWccBNTU3w8PBAamoq0tLSEB0dDZlMJnWZfeh0OrS2trrscivZp/v37+PBgwfYvHlzv02IAECpVGL27NkSVGYfeOYmm2AYsK36+noUFRWhqKgIdXV1cHNzQ0pKCtatW4fY2Fi7CwCv02q1AFz32ivZn7q6OuTn52Pq1KnIysqSuhy7xDM32URv8w3DwOj17ghYVFSE6upqqFQqJCcnY/ny5YiPj3eY5kytVgu1Wu2St2eR/TEYDDhw4AD8/PywZs0aqcuxWzxzk01wZWB0encELCoqQmVlJRQKBZKSkpCTk4MpU6Y45GYpbB4ke3Ly5Ek0NTVh3759lg22qD+euckmGAasN9iOgFu3bu2zI6Cj0mq1SElJkboMIjx8+BD37t3Dxo0bXfYuAWvxzE02wTAwtNHuCOhoOjs70dzczH4Bklx9fT1yc3ORmZmJqVOnSl2O3eOZm2yCYaA/vV6P0tJSFBYWWnYEjI2NHdGOgI6GzYNkDwwGAw4ePAgfHx/L4CsaGs/cZBMMAz26u7vx+PFjFBYW2mxHQEei1WqhUqkmbOwx0UBOnz6NhoYG7N271+Evu00U1z5zk824chgwGAx4+vQpioqKUFpaOi47AjoKrVYr+fRDcm1FRUW4c+cO1q9f75K7D46W6525aVz0hgFHuf1trHp3BCwqKkJJScmE7AjoCLRaLRISEqQug1xUY2Mjjh8/jvT0dEyfPl3qchwKwwDZhCusDNjLjoD2Sq/Xo7Gxkf0CJAmj0YiDBw/Cy8sL69ev5+rUCDnvmZsmlLOGAXvcEdBeVVdXA2DzIEnjzJkzqK2txccffwy1Wi11OQ7Huc7cJBmj0QhBEOx6TK617H1HQHul1WqhUCgGnPtONJ6Ki4tx+/ZtrFmzhmF0lBgGyCZ6dyx01DfJ3h0Be6cB2vOOgPaqt3nQGQIhOY6mpiYcO3YMqampmDVrltTlOCyGAbIJo9HokJcIamtrUVhYiKKiIjQ2Ntr9joD2TKvVYvLkyVKXQS7EZDLh4MGDcHd3x4YNGxjYx8Dxzt5klxwpDAy2I+DatWvtfkdAe9Xd3Y36+nrMmzdP6lLIhRQUFKC6uhp79uxxmimeUnGMszfZPaPRaNe3FTrLjoD2is2DNNFKS0tx48YNrFq1ChEREVKX4/AYBsgmxmNl4MWLFygtLcWqVatGtfznjDsC2iutVgu5XI7g4GCpSyEX0NzcjKNHjyIpKQlz5syRuhynwDBANmHrMPDixQt88803MJlMSE9PR2RkpFXPc/YdAe2VVqtFaGgoV1ho3JlMJhw6dAhqtRqbNm1in4CNMAyQTURFRdls452XL19agoBMJkNRUdGQYcBVdgS0Z1qt1urARjQW58+fR1VVFT766CO4u7tLXY7TYBggm8jMzIQoimM+TllZGb755huYzWYAPUN/Hj16hJUrV/b5BKDX61FSUoKioiLLjoCTJ0926h0B7ZXBYEBdXR1mz54tdSnk5J48eYLr169jxYoVDJ82xjBAo1bdYcTDBj1etRtQ32WCWQRkAhCkliPSS4nMQDeEeVj/EisvL8e//uu/wmQy9QkWOp0O5eXlCA8Pd+kdAe1VTU0NRFFk8yCNq9bWVhw5cgQJCQnIzs6WuhynwzBAI9bUZUJ+WRsqdEbIAJhf+55ZBGr1JtTrTbhbr0eUpwJrY7zhrx76WvKrV68GDAIAIAgCjh07hra2NpfeEdBeabVayGQyjmamcWM2m3Ho0CEoFAps3ryZfQLjgGGARqSoUY/88naYf3y/Ng/yuN6vv9IZsb+4CeuivZEaMPC88MrKSvzmN7+B0Wgc8FKDKIpobm7G4sWLkZGR4bI7AtorrVaLkJAQh5kzQY7nwoULqKiowO7du3kJcJxwugpZrahRjxNl7TCJgLXdASIAkwgcL2tDUaO+3/erqqqGDAKW44giIiMjGQTsUO8YYqLx8OzZM1y9ehVLlixBdHS01OU4LYYBskqj3oT88vYxHSO/vB1NXSbLv1dWVuLrr79Gd3f3sM2HMpkMhYWFY/r5ZHtGoxG1tbWYNGmS1KWQE2pra8Phw4cRHx+PBQsWSF2OU2MYIKucLG+zXBoYLbMI5Je1AejpQP/iiy9gMBise67ZDI1GA5PJNPyDacLU1tbCbDazeZBszmw24/Dhw5DJZHjrrbfYJzDOGAZoWNUdRlTojBABdLa14C/XTkPZg9uW71//bj8+/2QzGivL8Nm+TfjvOXH4x3cW9zuOCKBCZ0R1R8+AovT0dEyZMgURERH97hceaDvkrq4uPH/+fBx+QxotrVYLQRAQGhoqdSnkZC5fvoyysjJs3boVnp6eUpfj9NjxQ8N62KC33DXg7u2Lzf/lr3Hwv3+KP/j2Alpqq3Bh/9/i578+CTcvH6z8vf8T+vZWnPnnPxvwWLIfj7cyygtbtmzp873u7m40NzejsbERTU1NaGpqQkNDAxobG9Ha2gqz2Yzq6mokJCSM++9M1tFqtQgODuZoZ7KpFy9e4NKlS1i8eDF3wpwgDAM0rFfthj53DSTNX4ai83nI/7tfoqq0EMv/3S8QEBEDAJg8bS6e/3Bt0GOZAbzSDXxpQKVSISQkZMBb1MxmM9rb2+Ht7T2WX4VsTKvV8hIB2VR7ezsOHz6M2NhYLFy4UOpyXAYvE9Cw6rv6X6df90d/ikdnj0OhUmPOtg9Hdjz9yK/7y2Qy+Pj48LqhHTGZTKipqWEYIJsRRRFHjhyBKIrYsmULtxOfQPwvTUMSRXHAxsEX976HXKVCY2UZ9O1tIzqmWYRNRheTtOrq6mAymRgGyGauXLmC58+fY8uWLZwoOsEYBmhIgiBA9saH8Y6WJhz9sz/G+3/9NZIXLEf+3/23ER1TJmBUn/ANBgNDhB3RarUAwBkDZBNlZWW4ePEicnJyEBcXJ3U5Loc9AzSsILUcta8t7R/7i19g2pptiEqfjtD4JPzDjsV4/P0FJGYvse54boOPJjaZTGhubkZDQ4Plf/X19aivr4dOp8PSpUt5HdFOaLVaBAUFcVtoGjOdTodDhw4hOjoaixYtkrocl8QwQMOK9FKiXm+CGcCjguOoeVaCt//0nwAAKndPbP3l3+PAn/w+/sPvLuHv3l4AU3c39O2t+PPVmZi27m2s/vRPLMeSAYj07Ok8Ly0tRWNjI1paWtDQ0IC6ujq0trZaPv0LggBBECw7GAKAn5/fRP3aNAw2D5ItiKKIo0ePwmQyYevWrewTkAjDAA0rM9ANd+t7RglnLN+IjOUb+3w/bsZ8/CL/HgDg/zz1cMhjmX88ntlsxnfffTfkY0VR7HNZQCaTITExcRS/Adla722eKSkpUpdCDu7atWt4+vQp3nvvPd4tJCFGMBpWmIcCUZ4KjLWPXwAQ5alAmIcCMpkMGzdutLp3QBAExMfHQ60eeLMjmlj19fUwGo0cQ0xjUlFRgfPnz2P+/PmYMmWK1OW4NIYBssraGO9+jYQjJRN6jtNr2rRpeOedd6xaFhRFEWlpaWMrgGyGzYM0Vh0dHTh48CAiIyOxdOlSqctxeQwDZBV/tRzrose2hLcu2hv+6r7Ng4mJiVYHgvb2drS1jew2RhofWq0WAQEBcHNzk7oUckCiKOLYsWMwGAzsE7ATgsh7tWgEihr1yC9v75kVYMXjBfSsCKyL9kZqwOBL/E+fPsW3337bp1nwde7u7ujq6oLZbEZ0dDRSUlKQkpICX1/f0f0iNCZfffUVvL29sW3bNqlLIQf0/fff48yZM3j33XfZB2Qn2EBIVrt37x7CwsKwNyUE+WVtqNAZLXsWvKn365FeCqwdYEXgTVOmTMHOnTvx7bffDrgz4YoVK5CcnIzS0lIUFxejoKAAp0+fRkREBFJTU5GSkgJ/f39b/Jo0DFEUuUcEjdqrV69QUFCA7OxsBgE7wjBAVnny5AmOHz+OLVu2ICM8HO8l+qG6w4iHDXq80hl6bj0Ue1YBgtzkiPRUIjPQDWEe1r/E4uPjsXPnTvz2t7+F2Wzuc4thUlIS3N3dMXXqVEydOhV6vR6PHz9GcXExzp8/j7NnzyI8PBypqalITU1FQEDAeP2ncHkNDQ3o7u5m8yCNWGdnJw4ePIjw8HAsW7ZM6nLoNQwDNKyOjg4cP34c8fHxSE9Pt3w9zEOBMI+fRoaKojjmvQPi4uLw3nvv4ZtvvrFcMpg8eTI8PDz6PM7NzQ2ZmZnIzMxEV1cXnjx5guLiYly+fBnnzp1DaGioJRgEBQWNqSbqi82DNBqiKOL48ePo6urC7t27IZcPvVpIE4thgIYkiiJyc3NhMpmwadOmId/sbbWJUGxsLN5//33867/+K0wm07B3EajVaqSnpyM9PR0GgwFPnz6FRqPBtWvXcOHCBQQHB1uCQXBwMDc7GiOtVgtfX99+AY1oKLdu3UJJSQl27NjB4WF2iGGAhvTw4UMUFxdj27ZtEzoQZPLkyfjggw9w+fLlEQ22USqVluZCo9GIZ8+eQaPR4MaNG7h06RICAwORkpKC1NRUhIWFMRiMglar5SUCGpGqqiqcPXsWc+bMQXJystTl0AB4NwENqqWlBb/61a+QlJSEt956S+pyxsRoNOLFixfQaDQoKSmBXq+Hv7+/JRhMmjSJwcAKoijiL//yLzFv3jzk5ORIXQ45AL1ej88++wzu7u7Ys2cPLw/YKa4M0IB654Wr1WqsWbNG6nLGTKFQICEhAQkJCVi/fj1evnwJjUaD+/fv4/r16/D19bUEg8jISAaDQTQ1NaGrq4srA2QVURRx4sQJdHR04IMPPmAQsGMMAzSgGzdu4OXLl9i1a5fTDZaRy+WIj49HfHw81q1bh7KyMmg0Gjx69Ag3btyAt7e3JRhERUVxIMpr2DxII/HDDz9Ao9Hg7bff5q2/do5hgPqpra3FuXPnMHfuXMTGxkpdzriSyWSIjY1FbGws1qxZg4qKCmg0GhQXF+PWrVvw9PS0BIOYmBiXDwZarRbe3t7w8vIa/sHk0qqrq3H69GnMnDkTqampUpdDw2AYoD6MRiMOHz6MgIAAl7sPWCaTISYmBjExMVi9ejVevXplCQY//PADPDw8kJycjNTUVEyePNkllzzZPEjW6OrqwoEDBxAcHIxVq1ZJXQ5ZgWGA+rh48SLq6uqwd+9eKBSu+/IQBAFRUVGIiorCypUrUVVVZQkGd+/ehZubmyUYxMbGusR/K1EUodVqMXv2bKlLITsmiiLy8vLQ3t6OTz75xCX+bjgD/r9EFuXl5bh+/TqWLFmC8PBwqcuxG4IgICIiAhEREVi+fDlqamqg0WgsDYhqtRpJSUlISUnBlClTnPbk19rais7OTq4M0JDu3buHR48eYevWrQgMDJS6HLKSc561aMS6urpw5MgRREZGYv78+VKXY7cEQUBYWBjCwsKwZMkS1NXVWYLBw4cPoVKpkJiYiJSUFCQkJECpVEpdss1UVVUBAIMiDaqmpgYnT57E9OnT+0wrJfvHMEAAgNOnT6OjowO7du1y+SY5awmCgJCQEISEhGDx4sWoq6tDcXExNBoNCgsLoVQqkZCQgJSUFCQmJkKlUkld8photVp4enqyeZAG1N3djYMHDyIgIACrV6+WuhwaIYYBQklJCe7du4cNGzbw9p8xCA4ORnBwMHJyctDQ0IDi4mIUFxfj0KFDUCgUmDJliiUYOOLtmr3Ng5zBQAPJz89HS0sLPvnkE6daEXMVDAMuTqfT4cSJE0hMTMS0adOkLsdpBAYGYsGCBViwYAGamposweDIkSOQy+WIi4tDamqqZTdGe9fbPDh9+nSpSyE7dP/+fTx48ACbN2/mxmAOimHAhfVOBwOADRs28BPfOPH398e8efMwb948tLS0WILBsWPHLHMOUlNTkZycbLeb/7S3t0On07FfgPqpq6tDfn4+pk6diqysLKnLoVFiGHBh9+7dQ2lpKXbs2MHrwBPE19cXc+fOxdy5c9HW1mYJBrm5ucjNzcXkyZMtwcCe/j/pbR7knQT0OoPBgAMHDsDPz88pxpa7MoYBF9XU1ITTp09j6tSp3EVMIt7e3pg9ezZmz54NnU6HkpISaDQa5OfnIy8vDzExMUhNTUVKSsqE7hg5EK1WC3d3d/j4+EhaB9mXkydPoqmpCfv27XP4BllXxzDggsxmM44cOQIPDw92/doJT09PzJgxAzNmzEBHRwdKS0uh0Whw+vRpnDx5ElFRUZaxyL6+vhNeH5sH6U0PHz7EvXv3sHHjRoSEhEhdDo0Rw4ALun79OioqKvDRRx9BrVZLXQ69wcPDA9OmTcO0adPQ2dmJx48fQ6PR4Ny5czhz5gwiIiIswWCi7v7QarXIzMyckJ9F9q+hoQG5ubnIzMzE1KlTpS6HbIBhwMVUV1fjwoULmD9/PqKjo6Uuh4bh7u6OrKwsZGVloauryxIMLl68iIKCAoSHh1uCwXhNe2tvb0dbWxubBwlAz/4lBw4cgI+PD9atW8fVIifBMOBCejchCg4OxuLFi6Uuh0ZIrVYjIyMDGRkZ6O7uxpMnT1BcXIwrV67g/PnzCA0NtQSD4OBgm/3c3m2L2TxIAHDq1Ck0NDRg79697BNwIgwDLuTcuXNobGzk5iFOQKVSIS0tDWlpaTAYDHj69CmKi4tx/fp1XLx4EUFBQUhNTUVqaipCQkLG9OlNq9VCrVbDz8/Pdr8AOaSioiLcuXMH69evR2hoqNTlkA3xHcFFvHjxAjdu3MDKlSvZ7ONklEolUlJSkJKSAqPRiGfPnqG4uBg3b97E5cuXERAQYAkGYWFhIw4G1dXVCA8P53Kwi2tsbMTx48eRnp7O4VNOiGHABej1ehw9ehSTJ0/G3LlzpS6HxpFCoUBSUhKSkpJgMpnw/PlzaDQa3LlzB1evXoWfn58lGFh7d0BVVRVSU1MnoHqyV0ajEQcPHoSXlxfWr1/PYOiEGAZcwMmTJ9HV1YVNmzbxL7ELkcvlSEhIQEJCAkwmE8rKyizbLl+/fh0+Pj6WOQZRUVEDvjY6OjrQ0tLC5kEXd/bsWdTW1uLjjz/mHUhOimHAyRUVFeHhw4fYvHkzr/m6sN79EOLi4rB27VqUl5dbdle8ceMGvL29kZycjNTUVERHR1t2rmTzIBUXF+PWrVtYs2YNQ6ETYxhwYm1tbcjLy0NKSgrvEScLmUyGyZMnY/LkyVizZg0qKiqg0WhQXFyM27dvw9PT0xIMKisroVKpEBAQIHXZJIGmpiYcO3YMqampmDVrltTl0DgSRFEUpS6CbE8URfz2t79FdXU1fv7zn9vtBjhkP0RRRGVlpSUYNDc3Qy6Xw83NDZs3b0ZsbCzkcrnUZdIEMZlM+PLLL9HR0YGf/exnDrntNlmPYcBJ3b59G/n5+di5cycSEhKkLoccTO+Wxf/yL/8CURTR1dUFNzc3JCUlITU1FXFxcbw91cmdPn0at27dwscff8zLRC6Af5udUENDA86cOYOZM2cyCNCoCIKAgIAA6PV6bNq0CWFhYdBoNNBoNHjw4AHUajUSExORmpqK+Ph4KJVKqUsmGyotLcWNGzewatUqBgEXwTDgZHo3IfLx8cGKFSukLoccWHV1NQAgIiICwcHBCAsLw5IlS1BXV2e5lPDo0SMolUpLMJgyZQqn0jm4lpYWHD16FElJSZgzZ47U5dAEYRhwMleuXEFVVRX27NnDkzKNSVVVFRQKRZ89DwRBQEhICEJCQrB48WLU19ejuLgYGo0GBw4cgEKhQEJCAlJTU5GQkMDb0ByMyWTCwYMHoVareSuyi2EYcCKVlZW4dOkSFi5ciMjISKnLIQdXXV2NsLAwy22GAwkKCsLChQuxcOFCNDY2WoLBoUOHIJfLMWXKFKSkpCApKYkNaA7g/PnzqKqqwkcffQR3d3epy6EJxDDgJAwGA44cOYLw8HDk5ORIXQ45gaqqKsTFxVn9+ICAAMyfPx/z589Hc3OzJRgcPXoUMpkM8fHxSElJQXJyMt9o7NCTJ09w/fp1rFixgh8mXBDDgJM4e/YsWlpa8LOf/Yy3f9GYdXV1oaGhAfPnzx/V8/38/JCdnY3s7Gy0trZagsHx48eRm5uL2NhYSzDw9PS0cfU0Uq2trThy5AgSEhKQnZ0tdTkkAYYBJ/D06VPcvn0ba9asQVBQkNTlkBOoqakBAJtMnPPx8cGcOXMwZ84ctLe3o7i4GMXFxcjLy0NeXh4mT55s2WjJy8trzD+PRsZsNuPQoUNQKpXYvHkz+wRcFMOAg+vs7MSxY8cQHx/PCWFkM1VVVZDL5QgODrbpcb28vDBr1izMmjULOp0OJSUlKC4uxsmTJ5Gfn4+YmBhLMPDx8bHpz6aBXbx4ERUVFdi9ezeHk7kwhgEHJooi8vLyYDQasXHjRiZ6spnq6mqEhoaO6yUnT09PzJgxAzNmzEBHRwdKS0tRXFyMM2fO4NSpU4iMjLRspMR9NcbHs2fPcOXKFSxduhTR0dFSl0MSYhhwYIWFhSgqKsLWrVv5KYpsqqqqakLfHDw8PDBt2jRMmzYNer3eEgzOnTuHM2fOYNKkSZZg4Cr7JIiiiJs3byI+Pt7mKzRAz94lhw8fRnx8PBYsWGDz45NjYRhwUC0tLcjLy0NGRgbS09OlLoeciMFgQH19vWQDZ9zc3JCVlYWsrCx0dXXhyZMn0Gg0uHjxIgoKChAWFobU1FSkpqb2mYHgbOrr63H69GkAQFpaGhYtWmSzUGA2m3H48GHIZDK89dZbXFUkhgFHJIoijh07BpVKhTVr1khdDjmZmpoaiKJoF9vVqtVqpKenIz09Hd3d3Xj69Ck0Gg2uXLmC8+fPIyQkxBIMxuPTs73QaDQoKiqyWSi4fPkyysrKsGvXLt7NQQAYBhzSzZs38eLFC3zwwQe8X5tsrqqqCjKZDCEhIVKX0odKpbK88RsMBjx79gwajQbXr1/HxYsXERQUhJSUFKSmpiI0NNSpPu327idXVFSEoqIiJCYmws3NDa2trQgMDERERASSkpKsagB88eIFLl26hMWLF2Py5MnjXDk5CoYBB1NXV4eCggLMmTNnRANhiKyl1WoREhJi17sSKpVKJCcnIzk5GUajEc+fP4dGo8Ht27dx5coVBAQEWIJBeHi4UwUDmUwGf39/uLm5wWQyoby8HHfu3IFSqcT06dOxZMmSQcdAt7e34/Dhw4iNjcXChQsnuHKyZ/b7t536MZlMOHz4MPz9/bFs2TKpyyEnVV1dbReXCKylUCiQmJiIxMREmEwmvHjxAhqNBnfv3sW1a9fg5+dnCQYREREOEwze3F1eqVRi/vz5mDNnTr/RzjqdDrdu3cKNGzfw7Nkz7Nixo9/MEVEUceTIEYiiiC1btgw5ZppcjyC++Yoju3X+/Hlcu3YNe/fudaiTNTkOo9GIP//zP8fq1asdfm6F2WzGy5cvodFoUFJSAp1OBx8fH0swiIqKsutgcOPGDZw+fRpyuRwLFy4cMAS8qaGhAd999x2MRiM++eSTPpcRL1++jAsXLuCDDz7gqiL1w5UBB1FRUYGrV69i8eLFDAI0bmpra2E2m53iNSaTyRAXF4e4uDisXbsW5eXllka8mzdvwsvLyxIMoqOj7eqTcnt7Oy5fvozo6Gi8++67Vm/yFBgYiPfeew+fffYZjhw5gnfffReCIKCsrAwXL15ETk4OgwANiGHAAXR3d+PIkSOIiIjg/cA0rqqqqiAIAkJDQ6UuxaZkMhkmT56MyZMnY82aNaioqLDsl3D79m14enoiOTkZqampmDx5suTB4MaNGxBFEdu3bx/xbo9+fn7YsmULvvnmG5SUlCA6OhqHDh1CdHQ0Fi1aNE4Vk6NjGHAAp0+fRnt7O95//33JT1Lk3LRaLYKDg6FUKqUuZdwIgoDo6GhER0dj5cqVqKqqgkajgUajwZ07d+Du7m4JBrGxsZJs/PX8+XMkJiaO+ra/KVOmIDQ0FEVFRbh79y5MJhO2bt3K8wcNimHAzj1+/Bh3797F+vXrXWbyGknH0ZoHx0oQBERERCAiIgLLly9HdXW1JRjcu3cPbm5uSEpKQkpKCuLj4yfkDouOjg5otdoxD31KTU3F5cuXYTKZ8N5778Hb29tGFZIzYhiwYzqdDsePH0dCQgKmT58udTnk5EwmE2pqapCZmSl1KZIQBAHh4eEIDw/H0qVLUVtbawkGDx48gEqlsgSDKVOmjNvqycuXLwEAsbGxYzpOQEAATCYTkpOTMWXKFBtURs6MYcBOiaKI3NxciKLITYhoQtTV1cFkMrnUysBgevsmQkNDsWTJEtTV1UGj0aC4uBiPHj2CUqlEYmIiUlJSkJCQAJVKZbOf3dnZCQBj+iTf0dGBs2fPAgAHC5FVGAbs1IMHD1BSUoLt27dzj3eaEFqtFgAQFhYmcSX2Jzg4GIsWLcKiRYvQ0NBgCQYHDx6EQqFAQkICUlJSkJiYOOjAH2v1rjiYTKZhL0uIojjgB4XTp0/DYDBALpfzgwRZhWHADjU1NeHkyZOYOnUqUlJSpC6HXERVVRWCgoJs+inXGQUGBmLhwoVYuHAhmpqaLMHg8OHDkMvliI+PR2pqKpKSkkZ8JwDwUxgwGAz9wkB1hxEPG/R41W5AfZcJZhGQCUCQWo5ILyUyA90Q5qFAdHQ0ZsyYgd/85jc2+Z3J+TEM2Bmz2YyjR4/C3d0dq1evlrocciGu1jxoC/7+/pg/fz7mz5+P5uZmFBcXo7i4GEePHrXMOegNBtbsGwDAEsZ0Op1laFBTlwn5ZW2o0BkhA2B+7fFmEajVm1CvN+FuvR5RngqsTZ8Kb7kIs9nMlQGyCsOAnfn+++9RXl6O3bt3j3m5kchaZrMZ1dXVSE1NlboUh+Xn54fs7GxkZ2ejtbUVJSUl0Gg0OH78OARBQGxsLFJTU5GcnDzkLYNRUVFQKpUoLi7GwoULUdSoR355O8w/zoo1D/K83q+/0hmxv7gJ05RtEEWRPQNkFYYBO1JdXY3z589j3rx5iImJkbocciH19fUwGo1cGbARHx8fzJ49G7Nnz0Z7e7slGOTl5SEvLw8xMTGWYPBmo2DvXQuPHj1CQNosnChrH9HPFgGYROCHLi/4pcxw6q2dyXYYBuyE0WjEkSNHEBQUhCVLlkhdDrmY3uZBhgHb8/LywsyZMzFz5kzodDqUlpZCo9Hg1KlTyM/PR3R0NFJSUpCSkgJfX18AQEZGBopeVCCvrA3A6Jf59VNmo6nLBH/1xA9OIsfCjYrsxNmzZ3Hz5k3s27fP6UbBkv07efIknj59ik8//VTqUlxGZ2cnSktLUVxcjGfPnsFkMmHatGnYuHEjTCYT/vZKKUw+Qehsb8M/7liMd/78M8Rk9Wwedf27/Sg6n4slH/8RTv/T/4PuDh0gCEhesAKr/uBPLJMGBQCRngq8l+gn3S9KDoErA3bg5cuXuH79OpYvX84gQJJg8+DEc3d3x9SpUzF16lTo9Xo8fvzYcvdAXZcIk29Iz+O8fbH5v/w1Dv73T/EH315AS20VLuz/W/z81yehb2vBu3/+GQIiJ8PQpccXP9+Ke7m/w4yN7wLouWRQoTOiusOIMA+e7mlwfHVITK/X4+jRo4iJiUF2drbU5ZALEkURWq0WiYmJUpfistzc3PpMfnzYoO9z10DS/GUoOp+H/L/7JapKC7H83/0CARF9+4qUajeEJ6ajqaqiz9dlPx4vzIPzSmhw3LVCYqdOnUJnZyc2b97MTURIEg0NDTAYDFwZsCOv2g397hpY90d/ikdnj0OhUmPOtg/7PaetvgaF504gOWdln6+bAbzSGcavWHIKfPeRUHFxMR48eIA1a9bAz89P6nLIRbF50P7Ud5n6fe3Fve8hV6nQWFkGfXtbn+/p29vw6z98HzkfforI1Kn9j6fvfzyi1zEMSKS9vR0nTpxAcnIysrKypC6HXFhVVRX8/PwsA25IWqIoWmYK9OpoacLRP/tjvP/XXyN5wXLk/91/s3yvS9eOr35/B1IXr8bC938+4DHNYs9xiQbDMCABURRx/PhxyGQyrF+/nhPCSFJsHrQvgiBA9sYp4dhf/ALT1mxDVPp0rPnDX+LZ7at4/P0FdHW046vf347EeUuxdO9/HPSYMgE8z9CQ2EAogbt37+LJkyd49913h5xERjTeepsH58+fL3Up9JogtRy1Py7tPyo4jppnJXj7T/8JAKBy98TWX/49DvzJ72PGpp2oKLqH7s4OFJ3PBQBkLN+IJXv/qO/x3DhngIbGMDDBGhsbcfr0aUyfPp3d2yS5pqYmdHV1cWXAzkR6KVGvN8GMnjf3jOUb+3w/bsZ8/CL/HgBg+c/+eMhjyQBEeirHqVJyFrxMMIHMZjOOHDkCLy8vrFq1SupyiNg8aKcyA90G3YNgpMw/Ho9oKAwDE+jq1auorKzEW2+9xW1iyS5otVr4+PjwcpWdCfNQIMpTMYZBxD0EAFGeCg4comHxFTJBqqqqcOnSJSxYsABRUVFSl0MEoCcMcFVgYhiNRrS1taG9vb3fP5ubm1FdXY3AwEDs3bsXALA2xhv7i5tgGsNNADKh5zhEw2EYmAAGgwFHjhxBaGgoFi1aJHU5RAB+ah6cM2eO1KU4pWvXruHJkydobW2FTqdDd3d3v8fIZDKIomi57c/L66cpgf5qOdZFe+N4WVu/51lrXbQ3NykiqzAMTICCggI0Nzfjk08+gVzOv5hkH1paWtDZ2cmVgXHy/PlzlJWVDfkYs/mnzgBPT09s2bKlz/dTA9QQISK/vL1nVoAVP1dAz4rAumhvpAaoR1E5uSL2DIyz58+f49atW1i+fDn3FSe7wubB8bVmzZoRjRhft27dgL1EaQFu2Jvij0jPns9ugx2x9+uRXgrsTfFnEKAR4crAOOrs7MTRo0cRGxuL2bNnS10OUR9arRZeXl7w9uY15fEQFBSEBQsW4PLly0M+ThAExMXFITk5edDH+KvleC/RD9UdRjxs0OOVztBz66HYswoQ5CZHpKcSmYFubBakUeGrZhzl5+fDYDBg8+bNnP5FdofNg+Nv4cKFePToEZqbmwcdBywIAtauXWvVOSLMQ9Fn90FRFHluIZvgZYJxUlhYiMLCQqxduxY+Pj5Sl0PUhyiKqKqqYhgYZwqFAllZWUPuC7Bw4UIEBASM6vgMAmQrXBkYB62trcjLy0NaWhoyMjKkLoeon7a2NnR0dDAMjKOWlhacPHkSpaWl8Pb2Rnt7e59QIAgCfHx8sGDBAgmrJOrBMGBjoiji2LFjUCqVWLdundTlEA2IzYPjx2w249atW7hw4QJUKhXefvttxMTE4J/+6Z+g1+stjxNFEevXr4dCwdMwSY+vQhu7desWnj9/jvfff59bwpLd0mq18PDw4CUsG9NqtcjNzUVVVRVmzpyJZcuWwc2tZxTw6tWrcfToUQA9qwJJSUmYMmWKhNUS/YRhwIbq6+tRUFCA2bNnIz4+XupyiAbV2zzIa8620d3djQsXLuDmzZsIDg7Gnj17+k0azczMxP379/Hy5UvI5XKsXr1aomqJ+mMYsBGTyYTDhw/D19cXy5cvl7ocoiFptVpkZWVJXYZTePz4MfLz86HT6bB06VJkZ2cPOFxMEASsX78en332GZYsWQJfX18JqiUaGMOAjVy+fBk1NTX4+OOPoVRyu1CyX70z8dkvMDZtbW04deoUNBoN4uPj8eGHH8Lf33/I5wQGBuI//af/xHME2R2GARt49eoVrly5gkWLFmHSpElSl0M0JDYPjo0oivjhhx9w7tw5KBQKbNmyBenp6VZfcmEQIHvEMDBG3d3dOHLkCCZNmoSFCxdKXQ7RsLRaLdzc3ODn5yd1KQ6ntrYWJ06cwKtXrzBt2jSsWLGCjcLkFBgGxujMmTNoa2vDzp07RzSHnEgqbB4cOYPBgMuXL+P69esICAjA7t27ERMTI3VZRDbDMDAGT548wZ07d7Bu3ToEBgZKXQ6RVbRaLdLS0qQuw2E8e/YMeXl5aG1tRU5ODubPn8/ZAOR0+IoepY6ODhw/fhxTpkzBjBkzpC6HyCodHR1oaWlhv4AVdDodTp8+jUePHmHy5Ml47733GPrJaTEMjIIoisjNzYXJZMLGjRu53EoOg82DwxNFEffu3cPZs2chCAI2bdqErKws/j0np8YwMAoPHz5EcXEx3n77bW7/Sg5Fq9VCpVKNemMcZ1dfX4/c3FyUlZUhKysLK1asgKenp9RlEY07hoERam5uxsmTJ5GVlYXU1FSpyyEaETYPDsxoNOLq1au4evUqfH198cEHHyAuLk7qsogmDMPACIiiiKNHj8LNzY2jRMkhabVaJCUlSV2GXXn58iVyc3PR1NSE+fPnIycnhw2C5HL4ih+B77//HmVlZfjwww8tm48QOYrOzk40NTWxX+BHHR0dOHv2LO7fv4+oqChs374dISEhUpdFJAmGASvV1tbi/PnzyM7OxuTJk6Uuh2jEqqurAbB5UBRFPHr0CKdPn4bZbMb69esxffp0Xjohl8YwYAWj0YjDhw8jMDAQS5culbocolHRarVQKpUufXtcY2Mj8vLy8Pz5c6Snp2PVqlXw8vKSuiwiyTEMWOHixYuoq6vDvn37eC2RHJZWq0VYWJhLTso0mUy4fv06Ll++DC8vL+zcuRMJCQlSl0VkN/jONoyysjJcu3YNy5YtQ1hYmNTlEI2aVqtFfHy81GVMuPLycuTm5qK+vh7Z2dlYtGgRVCqV1GUR2RWGgSF0dXXh6NGjiI6Oxrx586Quh2jUurq60NDQgAULFkhdyoTR6/UoKCjAnTt3EBERgU8++YSBnmgQDANDOHXqFDo6OrBr1y6XXFol5+FKzYOiKKKoqAinTp2CwWDAmjVrMHPmTP4dJhoCw8AgSkpKcP/+fWzcuBH+/v5Sl0M0JlqtFgqFAsHBwVKXMq6am5uRn5+PJ0+eICUlBatXr4aPj4/UZRHZPYaBAbS3t+PEiRNISkrC1KlTpS6HaMy0Wi1CQ0Od9tOx2WzGjRs3cPHiRbi7u+Odd97hcCWiEWAYeIMoijhx4gQEQcCGDRt47zE5Ba1Wi5iYGKnLGBeVlZU4ceIEamtrMXv2bCxZsgRqtVrqsogcikuGgY6ODnz22WeYOnUqFi5cCLlcbvnevXv38PjxY7zzzjvcoIScQnd3N+rr6zF37lypS7Gprq4unD9/Hrdu3UJYWBj27t2LSZMmSV0WkUNyyDBw4sQJVFVVYfHixUhMTBzxp/fq6mq0tLTg0qVLePz4MbZu3YrAwEA0Njbi1KlTmDZtGpcYyWnU1NRAFEWnah4sKSlBfn4+9Ho9Vq5ciTlz5jjtJRCiieCQYaC2thbV1dX47rvvEBoaiiVLlowoFDQ2Nlr+XF1djV/96ldYuXIlHj16BC8vL6xatWq8SieacFqtFnK53Cnm7re2tiI/Px+lpaVISEjA2rVr4efnJ3VZRA7PIcPA62pra/Hdd98hMDAQc+fORVZWFmpra+Hh4QE/P78BA0JDQwNkMhnMZjNEUYTJZMLJkycBADt27OD1RnIqWq0WISEhfS6HORqz2Yzbt2/j/PnzUKlUePvtt5GSksKeHiIbcfgwIIoigJ43+Ly8PFRWVuL+/fsAAA8PD8TGxiI7OxsRERGW5zQ0NMBsNg94vKNHj2LDhg1IS0sb99qJJoJWq+3z+nc01dXVlkuDM2fOxLJly7hrKJGNOWQY6A0ArwsNDUVOTg6SkpIwd+5ctLS0oLKyEoWFhdi/fz/S09Oxfv16qNVq1NfXD3rsrq4uHDx4EKWlpVi7di1POuTQjEYj6urqMHPmTKlLGbHu7m5cvHgRN27cQHBwMPbs2YOoqCipyyJySg4ZBlpaWix/jo6OxtKlS/vcNhUaGorQ0FAkJiZi0aJFePToEfLz8/HFF19g9+7dfZ4/mMLCQrx48QJ79+6Fr6/vuPweROOtpqYGZrPZ4ZoHnzx5gry8POh0OixduhTZ2dkOfZmDyN45XBh4+fIl2tvb4e/vj02bNg1777RMJkNWVhYmTZqEL7/8EgcPHhz0EsHrRFG09BMQOSqtVguZTIbQ0FCpS7FKW1sbTp06BY1Gg7i4OOzatQsBAQFSl0Xk9BwqDJhMJhw/fhzR0dHYvXv3iJqHgoOD8dZbb+Hbb78d8nGCIEChUCAnJwdz5syBUqkca9lEktFqtQgODrb7rbdFUcSdO3dQUFAAuVyOLVu2ID09nQ2CRBPEvs8Qb6ioqEBTUxO2bds2qpNEYmIiIiMj8erVq37f671HedasWcjJyYGHh8eY6yWSmlartftLBLW1tcjNzUVFRQWmTZuGFStWwN3dXeqyiFyKQ4WBZ8+ewcPDY0wnNy8vrz7/LggCRFFEcnIyli1bxiVJchomkwm1tbV2u7+GwWDA5cuXcf36dQQEBGD37t1OOzKZyN45VBh48eIF4uLixrR02NnZ2effIyMjsWrVKoe+9YpoILW1tTCZTHa5MvD8+XPk5uaitbUVOTk5mD9/vt1fyiByZg7zt08URVRVVSEzM3NMx6mqqgIAyOVybN++HQkJCbwuSU5Jq9VCEAS7ah7U6XQ4c+YMHj58iMmTJ2Pnzp0ICgqSuiwil+cwYUAQBMv/hiOK4qCPi4+PR2trKwwGAxITE21dJpEkzGZzv9n8Wq0WQUFBUKlUElX1E1EUcf/+fZw9exYAsGnTJmRlZTGIE9kJhwkDAKBUKmEwGPp9vbrDiIcNerxqN6C+ywSzCMgEIEgtR6SXEpmBbgjz6PlVd+zYgVOnTuHFixcTXT7RuCguLsaBAwfg5+eHyMhIhIeHIzw8HJWVlXZxiaC+vh65ubkoKytDZmYmVq5cyR1BieyMQ4eBpi4T8svaUKEzQgbg9ekBZhGo1ZtQrzfhbr0eUZ4KrI3xhr+ag0vIuXh4eEAURTQ1NaG5uRmFhYWWKZ1NTU04ePCgJSCEh4dPWKe+0WjE1atXcfXqVfj4+OCDDz5AXFzchPxsIhoZhwoDfn5+qK6uBgAUNeqRX94O84+TiQcbI9T79Vc6I/YXN2FdtDfq6+v5yYQcjk6ng1Kp7LfsHxkZCYVCAaPR2G9Ut16vh0ajgUajsXxv2rRp2Lhx47jW+vLlS+Tm5qKpqQnz5s1DTk4OZ3YQ2TGHCgNpaWkoKCjAveo2nNZ2jei5IgCTCBwva4OxS4FVmfHjUyTROPn666/R1taGBQsWYPbs2ZZQIJfLERcXhydPngy4b8ebXwsODh63Gjs7O3HmzBncv38fUVFR2L59u1NsnUzk7BwqDKSnp+PMtZs4U6UHRt14JEKetRjhcRwqRI5Fr9ejq6sL586dw9WrV/uEgilTpuDx48dDPl8QBCQmJmLu3Lk2r00URTx69AinT5+GyWTCunXrMGPGDDYIEjkIhwoDXl5e8JqzBt0QAYz2JCNAkMlwqd6M9zhfiBxUbyi4dOkS0tPTkZSUNOTjBUGAr68vNm/ebPM36MbGRuTl5eH58+dIS0vDqlWr4O3tbdOfQUTjy6HCQHWHEd2eA7+Df/df/h38wyKx6tP/avna13/wLuJmLkDOrn/f98GCDBU6I6o7jJa7DKzR0dGB+/fv44cffsCkSZOwbdu2Uf0eRLZiNBrx6NGjYZsCZTIZ3nnnHZtuyW0ymXD9+nVcvnwZnp6e2LlzJxISEmx2fCKaOA4VBh426PvdNdBr0y/+Av/4zhKkLl2HqLRpuH30X9HV3oYF7/98wGPJfjxemIfXgN/vJYoiysrK8MMPP6C4uNiymyG3NaaJ9ua1f5VKhYULF1ouFRgMBty5c2fAvoENGzbYdPhQRUUFcnNzUVdXh7lz52Lx4sV2Mc+AiEbHocLAq3bDoHcNuPv4YfN/+Wsc/OWn+OBvf42CX/0lfvbFiX6DWHqZAbzS9Z9Z0Ov1VYCmpibIZDLL1seCIPTb44BoPJlMJssobZVKhZycHMyaNavPG3B8fDx++OGHfs9VKpWYMmWKTerQ6/UoKCjAnTt3MGnSJHzyyScICwuzybGJSDoOFQbqu0xDfj9p/jJoLp7EP7+/Eqv/4L8hIHLy0MfT9z3eYKsAACxBAOgJA9zVkCbS5cuXYTabMXfuXCxZsmTAT+GxsbGWjbeAntdpUFAQOjo6cPDgQXzwwQeDhuPhiKIIjUaDU6dOobu7G6tXr8asWbNGfTwisi8OEwZEUbTMFBhKzq5/jwenDmHOtg+HfaxZ/GnpNS8vD8XFxejo6OhzQh0M5xTQRGlqasLVq1exaNEiLF68eNDHqdVqREREWLboVqlU2LlzJ5qbm/Gb3/wG33//PebPnz/in9/c3Iz8/Hw8efIEycnJWLNmDXx8fEb76xCRHXKYMCAIAmQChg0EMrkcgmDdpxWZ0HPc7u5u3L171xIAhgsCZrMZNTU1ePToETw8PODp6QlPT094eHhALueEQ7Ktx48fQxAEzJs3b9jHJiQkWMLAtm3b4OfnBz8/P2RkZOD+/fsjCgNmsxk3btzAxYsX4ebmhh07diA5OXnUvwcR2S+HCQNAz14DtfqhLxWM6HhuPW/cKpUKf/iHf4iTJ0+ipKTEqpWBkpISaDSafl9Xq9V9wsGbYeHNPzM80HCeP3+OqKgoqxr0oqOjAQCZmZl9+gRSU1Px8OFD1NbWWjUEqKqqCidOnEB1dTVmz56NpUuXQq1Wj/6XICK75lBhINJLiXq9adAmwpGQAYj0/Gk8qo+PD3bs2GHZZ72pqWnI53/44YcICwtDR0cHdDoddDpdvz93dHSgpqbG8u8DbbLUGx56w8Gb/3zza9zz3bWYTCa8fPkSCxYssOrxvdsCv9kwGB8fD7VaDY1GM2QY6OrqwoULF3Dr1i2EhoZi7969iIiIGNPvQET2z6HeWTID3XC3Xj/kY/wnReOXl58Neyzzj8d7U1xcHH7v934P33//PS5dutTTq2DuHz88PDygUqmgUqng5+dnVf3d3d2WwDBQcNDpdKipqbH8ebDw8GZQGGr1geHBsbW1taG7u3tEHfsD3euvUCgQHR2NqqqqQZ9XUlKCkydPorOzE8uXL8fcuXPZIEjkIhzqnSLMQ4EoTwVe6YywopdwUAKASE/FoAOHFAoFFi5ciIyMDEvj1JuXDkbTQDjS8GAwGAYMDq//s7a21vK9gcKDSqUa0WULhgf70ru5z0CB9E2iKA45XVAul8Nk6n+ZrbW11XKJLCEhAWvXrrX6NUpEzsHhzvxrY7yxv7gJpjGkAZnQc5zh+Pn5YefOnSgtLUVeXh7a29stJ1xbTnIbjFKptDSAWcNgMAx72aKurg4vX75ER0cHuru7+x2jNzxYe9mCO9GNr97/vgMFveoOIx426PGq3YD6LhPMYs9rO0gtR6SXEpmBbkNO2DSbzbh9+zbOnz8PlUqFbdu2ITU1lfsJELkghwsD/mo51kV743hZ26iPsS7aG/5q6xv3kpKSEBcXh8uXL+P69etwd3e3yxOmUqmEr6+v1dMRXw8Pg60+1NXVoaysDDqdbtDwYG1w8PT0ZHgYod7/Xl1dP+3S2dRlQn5ZGyp0xn4TOc0iUKs3oV5vwt16PaI8FVgb0/N6f31lq7q6Grm5uaisrMSMGTOwfPnyCQm4RGSfBHG4tnk7VdSoR355e8+sACseL6DnU9O6aG+kBoy+K7qhoQGdnZ2IjIwc9TEcldFoHPayxev/fP0NrJdSqRw2MLz+PYYH4Fe/+hVCQkKwdevWUb/u10R54vTX/xupqalQKBT4/vvvERQUhA0bNiAqKmq8fwUisnMOtzLQKy3ADZM8lYN+QurV+/VILwXWjnBFYCCBgYFjer4jUygUI1p5MBqNwwaH+vr6YcPDSBomnXE+fnp6Oq5cuYIHtTqcrOwc0XNFACYRyC1vh8E3HABw69YtLFmyBPPmzeOtrUQEwIFXBl5nuXaqM/Tceth77dRNjkjP4a+dkn14PTwMd/lCp9MNGB4UCoVVKw6vX7awx0s+r2tubsY/7v8absveg3m0W3eLIiCa8VGiL7zlZo7TJqI+nCIMvGm4rmpyDr3hwdrLFnp9/9tSe8ODNasPvSsPUry2/uZKKQwe/sBYbvUTRUR5KfFeop/N6iIi5+CUH5cZBFyDQqGAj4+P1XPyTSbTsMGhsbERFRUVQ4aHkTRM2iI8VHcYYfAa+PLUX66bDoVKBYXKDd2dOoTGJ2PR7k8RkzW7/4MFARU6I6o7jFwpI6I+eEYglyGXy+Ht7Q1v7+FvKwX6hofBQkRTUxMqKyuh0+kGDA9yudzqlYfBwsPDBv2gPTEA8O5ffI5JSRkAgMJzufj603fx0T//G6IzZvR7rOzH44V5cAtuIvoJwwDRIEYbHoZafWhubh42PPSGg5ycHKSkpOBVu8HqEdzpy9bjVdFdXPmXf8Z7f/Vlv++bAbzS9Z9ZQESujWGAyEZGEx46OzsHHRLVO+Wyvmtkm3NFpc9A8aXTg36/3oabfRGRc2AYIJKIXC6Hl5cXvLwGX7IXRXHYbbsHes5QzCKbbImoL+5CQmTHBEGAbITv2a809xA6JXnQ78sENtkSUV8MA0R2LmgEg7I0F0/i5oGvseD93xv8eG4cNEREffEyAZGdi/RS9gzTGuT73/7nfT/dWhiXhN3/69sB7yQAetJ/pCdHPBNRX045dIjImVR3GPF1abPNjrc7yY9zBoioD14mILJzYR4KRHkqRjuI2EIAEOWpYBAgon4YBogcwNoY7xE3Er5JJvQch4joTQwDRA7AXy3HuuixvZGvs8GunUTknLheSOQgUgPUECEiv7y9Z1aAFc8R0LMisC7aG6kB6vEukYgcFBsIiRxMU5cJ+WVtqNAZB92zoPfrUV4KrOWKABENg2GAyEFVdxjxsEGPVzpDz62HYs8qQJCbHJGeSmQGurFZkIiswjBA5CQ4YpiIRosNhEROgkGAiEaLYYCIiMjFMQwQERG5OIYBIiIiF8cwQERE5OIYBoiIiFwcwwAREZGLYxggIiJycQwDRERELo5hgIiIyMUxDBAREbk4hgEiIiIXxzBARETk4hgGiIiIXBzDABERkYtjGCAiInJxDANEREQujmGAiIjIxTEMEBERuTiGASIiIhfHMEBEROTiGAaIiIhcHMMAERGRi2MYICIicnEMA0RERC6OYYCIiMjFMQwQERG5OIYBIiIiF8cwQERE5OIYBoiIiFzc/w/6b2lSsoqK1AAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'A': -inf, 'C': 2, 'B': 3, 'D': -inf, 'X1': 3, 'Y': 4, 'X2': 2}\n"
     ]
    }
   ],
   "source": [
    "G = nx.DiGraph()\n",
    "G.add_edges_from(\n",
    "        [('A', 'A'),\n",
    "         ('A', 'C'),\n",
    "         ('A', 'B'),\n",
    "         ('A', 'D'),\n",
    "         ('A', 'X1'),\n",
    "         ('D', 'D'),\n",
    "         ('D', 'Y'),\n",
    "         ('D', 'X2'),\n",
    "         ('X2', 'C'),\n",
    "         ('X2', 'X1'),\n",
    "         ('X2', 'X2'),\n",
    "         ('X1', 'B'),\n",
    "         ('X1', 'Y'),\n",
    "         ('Y', 'Y')])\n",
    "\n",
    "\n",
    "# Draw the SCG\n",
    "pos = nx.kamada_kawai_layout(G)  # Calcul automatique des positions\n",
    "nx.draw_networkx_nodes(G, pos, node_color='skyblue', node_size=200)\n",
    "nx.draw_networkx_labels(G, pos, font_size=8, font_color='black')\n",
    "\n",
    "bidirectional_edges = {(u, v) for u, v in G.edges if (v, u) in G.edges}\n",
    "for u, v in bidirectional_edges:\n",
    "    if u < v:  \n",
    "        nx.draw_networkx_edges(G, pos, edgelist=[(u, v)], arrowstyle='-|>', arrowsize=20, edge_color='gray', connectionstyle='arc3,rad=0.2')\n",
    "        nx.draw_networkx_edges(G, pos, edgelist=[(v, u)], arrowstyle='-|>', arrowsize=20, edge_color='gray', connectionstyle='arc3,rad=0.2')\n",
    "        \n",
    "normal_edges = [(u, v) for u, v in G.edges if (u, v) not in bidirectional_edges]\n",
    "nx.draw_networkx_edges(G, pos, edgelist=normal_edges, arrowstyle='-|>', arrowsize=20, edge_color='gray')\n",
    "\n",
    "\n",
    "self_loops = [(u, v) for u, v in G.edges if u == v]\n",
    "nx.draw_networkx_edges(G, pos, edgelist=self_loops, arrowstyle='-|>', connectionstyle='arc3,rad=5', arrowsize=10, edge_color='gray')\n",
    "\n",
    "plt.axis('off')\n",
    "plt.show()\n",
    "\n",
    "Interventions = [ ('X1', 4), ('X2', 3)]\n",
    "print(compute_t_NC(G, Interventions))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Computation of $t^{NC}_{V_{t_v}}$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "def calculate_t_VNC(P, gamma_s, t_NC, Interventions,  V, gamma_v):\n",
    "    if (P, gamma_s) == (V, gamma_v):\n",
    "        gamma_s += 1\n",
    "    if gamma_s > t_NC[P]:\n",
    "        return(math.inf)\n",
    "    \n",
    "    Interventions_gammas = {g for node, g in Interventions if (node == P and g >= gamma_s)}\n",
    "    gamma_prime = gamma_s\n",
    "    while 1:\n",
    "        if gamma_prime not in Interventions_gammas:\n",
    "            return(gamma_prime)\n",
    "        gamma_prime += 1\n",
    "    return(math.inf)  \n",
    "    \n",
    "\n",
    "\n",
    "def compute_t_VNC(G, V, gamma_v,t_NC, Interventions,check_directed=False):\n",
    "    \"\"\"\n",
    "    Compute (t^NC_{V_{t- gamma_v}}(S))_{S ∈ V^s}.\n",
    "    \n",
    "    Parameters:\n",
    "    - G: NetworkX DiGraph representing the SCG.\n",
    "    - V: the target node V for which t^NC is computed.\n",
    "     - Interventions: list of tuples [(V, gamma)], representing the interventions.\n",
    "    - gamma_v: the gamma value of V: t_v = t-gamma_v.\n",
    "    - t_NC: dict with nodes as keys and their t - t_NC values as values.\n",
    "    \n",
    "    Returns:\n",
    "    - t_VNC: dict with nodes as keys and their t - t^NC_{V_{t- gamma_v}}(S) values as values.\n",
    "    \"\"\"\n",
    "    # Priority queue and initialization\n",
    "    Q = []\n",
    "    heapq.heappush(Q, (gamma_v, V))  # Min-heap by using gamma\n",
    "    t_VNC = {node: math.inf for node in G.nodes}  # Initialize all t^NC to -infinity\n",
    "    seen = {node: False for node in G.nodes}  # Track if a node has been processed\n",
    "\n",
    "    # Main processing loop\n",
    "    while Q:\n",
    "        gamma_s, S = heapq.heappop(Q)  # Extract node with the smallest gamma\n",
    "        \n",
    "        # Process all unseen parents of S\n",
    "        for P in G.predecessors(S):  # Access parents using NetworkX\n",
    "            if not seen[P]:\n",
    "                if check_directed and gamma_s == 0 and (P, gamma_s) in Interventions:\n",
    "                    raise ExistsDirectedBackdoorPath(f\"A directed backdoor path from {P}_{{{gamma_s}}} exists.\")\n",
    "                # Compute t^NC for P\n",
    "                gamma_NC_P = calculate_t_VNC(P, gamma_s, t_NC, Interventions, V, gamma_v)\n",
    "                \n",
    "                if gamma_NC_P != math.inf:  \n",
    "                    heapq.heappush(Q, (gamma_NC_P, P))  # Add parent to the queue\n",
    "                    t_VNC[P] = gamma_NC_P  # Update t^NC value for P\n",
    "                \n",
    "                seen[P] = True  # Mark parent as seen\n",
    "\n",
    "    return t_VNC\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Implementation of Algorithm 3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [],
   "source": [
    "#For printing the results\n",
    "def formated(dic):\n",
    "    return({key: f\"{-value}\" if abs(value) == math.inf else f\"t - {value}\" for key, value in dic.items()})\n",
    "\n",
    "\n",
    "def is_ICA(G,Interventions):\n",
    "    t_NC = compute_t_NC(G, Interventions)\n",
    "    print(\"t_{NC} is:\",formated(t_NC))\n",
    "    \n",
    "    t_VNC = {}\n",
    "    try:\n",
    "        t_VNC[('Y', 0)] = compute_t_VNC(G, 'Y', 0, t_NC, Interventions, check_directed=True)\n",
    "    except ExistsDirectedBackdoorPath as e:\n",
    "        print(e)\n",
    "        return(False)\n",
    "    print(\"t^{NC}_{Y_0} is:\", formated(t_VNC[('Y', 0)] ))\n",
    "    \n",
    "    for (X, gamma_x) in Interventions:\n",
    "        t_VNC[(X, gamma_x)] = compute_t_VNC(G, X, gamma_x, t_NC, Interventions)\n",
    "        print(f\"t^{{NC}}_{{{X}_{{t-{gamma_x}}}}} is:\", formated(t_VNC[(X, gamma_x)]))\n",
    "\n",
    "    for F, X in itertools.product(G.nodes(),Interventions):\n",
    "        if t_NC[F] >= t_VNC[X][F] and  t_NC[F] >= t_VNC[('Y',0)][F]:\n",
    "            print(t_VNC[X][F], t_VNC[('Y',0)][F])\n",
    "            print(f\"Fork {F} at time t - {t_NC[F]}.\")\n",
    "            return(False)\n",
    "    return(True)\n",
    "        \n",
    "\n",
    "    \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Interventions: [('X1', 't - 4'), ('X2', 't - 3')]\n",
      "t_{NC} is: {'A': 'inf', 'C': 't - 2', 'B': 't - 3', 'D': 'inf', 'X1': 't - 3', 'Y': 't - 4', 'X2': 't - 2'}\n",
      "t^{NC}_{Y_0} is: {'A': '-inf', 'C': '-inf', 'B': '-inf', 'D': '-inf', 'X1': 't - 0', 'Y': 't - 1', 'X2': 't - 0'}\n",
      "t^{NC}_{X1_{t-4}} is: {'A': '-inf', 'C': '-inf', 'B': '-inf', 'D': '-inf', 'X1': '-inf', 'Y': '-inf', 'X2': '-inf'}\n",
      "t^{NC}_{X2_{t-3}} is: {'A': '-inf', 'C': '-inf', 'B': '-inf', 'D': '-inf', 'X1': '-inf', 'Y': '-inf', 'X2': '-inf'}\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "True"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Interventions\n",
    "Interventions = [('X1', 4), ('X2', 3)]\n",
    "print(\"Interventions:\", [(X,  f\"t - {gamma_x}\") for X, gamma_x in Interventions])\n",
    "is_ICA(G,Interventions)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Implementing the speed up"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [],
   "source": [
    "def calculate_max_t_VNC(P, gamma_s, t_NC, Interventions):\n",
    "    if gamma_s > t_NC[P]:\n",
    "        return(math.inf)\n",
    "    \n",
    "    Interventions_gammas = {g for node, g in Interventions if (node == P and g >= gamma_s)}\n",
    "    gamma_prime = gamma_s\n",
    "    while 1:\n",
    "        if gamma_prime not in Interventions_gammas:\n",
    "            return(gamma_prime)\n",
    "        gamma_prime += 1\n",
    "    return(math.inf)  \n",
    "\n",
    "def exists_fork_paths(G,t_NC, Interventions, t_VNCY):\n",
    "    \"\"\"\n",
    "    Compute (t^NC_{V_{t- gamma_v}}(S))_{S ∈ V^s}.\n",
    "    \n",
    "    Parameters:\n",
    "    - G: NetworkX DiGraph representing the SCG.\n",
    "    - Interventions: list of tuples [(V, gamma)], representing the interventions.\n",
    "    - t_NC: dict with nodes as keys and their t - t_NC values as values.\n",
    "    \n",
    "    Returns:\n",
    "    - True if and only if there isa collider-free-backdoor path with a fork\n",
    "    \"\"\"\n",
    "    \n",
    "    # Priority queue and initialization\n",
    "    Q = [(gamma_x, X) for X, gamma_x in Interventions]\n",
    "    heapq.heapify(Q)\n",
    "    \n",
    "    seen = {node: False for node in G.nodes}  # Track if a node has been processed\n",
    "\n",
    "    # Main processing loop\n",
    "    while Q:\n",
    "        gamma_s, S = heapq.heappop(Q)  # Extract node with the smallest gamma\n",
    "        if ((S,gamma_s) not in Interventions)  and  t_NC[S] >=  t_VNCY[S]:\n",
    "            #print(f\"Fork {S} at time t - {t_NC[S]}.\")\n",
    "            return True\n",
    "        \n",
    "        # Process all unseen parents of S\n",
    "        for P in G.predecessors(S):  # Access parents using NetworkX\n",
    "            if not seen[P]:\n",
    "                # Compute max t^NC for P\n",
    "                gamma_NC_P = calculate_max_t_VNC(P, gamma_s, t_NC, Interventions)\n",
    "                \n",
    "                if gamma_NC_P != math.inf:  \n",
    "                    heapq.heappush(Q, (gamma_NC_P, P))  # Add parent to the queue\n",
    "            \n",
    "                seen[P] = True  # Mark parent as seen\n",
    "\n",
    "    return False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "def fast_is_ICA(G,Interventions):\n",
    "    t_NC = compute_t_NC(G, Interventions)\n",
    "    try:\n",
    "        t_VNCY = compute_t_VNC(G, 'Y', 0, t_NC, Interventions,check_directed=True)\n",
    "    except ExistsDirectedBackdoorPath as e:\n",
    "        return(False)\n",
    "    \n",
    "    return(not exists_fork_paths(G,t_NC, Interventions, t_VNCY))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Interventions: [('X1', 't - 2'), ('X2', 't - 1')]\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "True"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Interventions = [('X1', 2), ('X2', 1)]\n",
    "print(\"Interventions:\", [(X,  f\"t - {gamma_x}\") for X, gamma_x in Interventions])\n",
    "fast_is_ICA(G,Interventions)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Checking that the fast implementation is consistent with the slow implementation on random graphs."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [],
   "source": [
    "import random\n",
    "import sys,os\n",
    "from contextlib import contextmanager\n",
    "random.seed(42) # For reproducibility\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "All tests passed. Both functions return identical results.\n"
     ]
    }
   ],
   "source": [
    "\n",
    "# Context manager to disable print\n",
    "@contextmanager\n",
    "def suppress_print():\n",
    "    original_stdout = sys.stdout\n",
    "    sys.stdout = open(os.devnull, 'w')\n",
    "    try:\n",
    "        yield\n",
    "    finally:\n",
    "        sys.stdout.close()\n",
    "        sys.stdout = original_stdout\n",
    "\n",
    "# Generate a random directed graph\n",
    "def generate_random_graph(num_nodes, edge_probability):\n",
    "    G = nx.DiGraph()\n",
    "    nodes = [\"V\" + str(i) for i in range(num_nodes - 1)] + [\"Y\"]  # Add node \"Y\"\n",
    "    G.add_nodes_from(nodes)\n",
    "    \n",
    "    # Add edges progressively\n",
    "    for _ in range(int(edge_probability * num_nodes**2)):\n",
    "        source, target = random.sample(nodes, 2)  # Pick distinct nodes\n",
    "        G.add_edge(source, target)\n",
    "\n",
    "    return G\n",
    "\n",
    "# Generate random interventions\n",
    "def generate_random_interventions(G, num_interventions, intervention_on_y_probability):\n",
    "    nodes = list(G.nodes)\n",
    "    interventions = []\n",
    "    for _ in range(num_interventions):\n",
    "        node = random.choice(nodes)\n",
    "        gamma = random.randint(0, 5)  # Random gamma value\n",
    "        interventions.append((node, gamma))\n",
    "    # Add one intervention explicitly involving \"Y\"\n",
    "    if random.random() < intervention_on_y_probability:\n",
    "        gamma_Y = random.randint(1, 5)\n",
    "        interventions.append(('Y', gamma_Y))\n",
    "    return interventions\n",
    "\n",
    "# Test is_IBC and fast_is_IBC\n",
    "def test_functions(num_graphs, num_interventions, num_nodes, edge_probability, intervention_on_y_probability):\n",
    "    all_results_match = True\n",
    "    for _ in range(num_graphs):\n",
    "        # Generate random graph and interventions\n",
    "        G = generate_random_graph(num_nodes, edge_probability)\n",
    "        Interventions = generate_random_interventions(G, num_interventions, intervention_on_y_probability)\n",
    "        # Evaluate both functions\n",
    "        with suppress_print():  # Suppress print outputs\n",
    "            result1 = is_ICA(G, Interventions)\n",
    "            result2 = fast_is_ICA(G, Interventions)\n",
    "        \n",
    "        # Check if results match\n",
    "        if result1 != result2:\n",
    "            print(f\"Mismatch found!\\nGraph: {G.edges}\\nInterventions: {Interventions}\")\n",
    "            print(f\"is_ICA: {result1}, fast_is_ICA: {result2}\")\n",
    "            all_results_match = False\n",
    "            break\n",
    "\n",
    "    if all_results_match:\n",
    "        print(\"All tests passed. Both functions return identical results.\")\n",
    "    else:\n",
    "        print(\"Test failed. Some results do not match.\")\n",
    "\n",
    "\n",
    "\n",
    "test_functions(\n",
    "    num_graphs=10000,            # Number of random graphs to generate\n",
    "    num_interventions=10,      # Number of interventions per graph\n",
    "    num_nodes=8,              # Number of nodes per graph\n",
    "    edge_probability=0.3,     # Probability of an edge existing between nodes\n",
    "    intervention_on_y_probability=.5\n",
    ")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Testing on a huge random graph. \n",
    "\n",
    "This graph is designed to be ICA by thanks to its ClusterDAG representation G"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Intra cluster edges done\n",
      "Inter cluster edges done\n",
      "self loop done\n",
      "done!\n",
      "Nb of nodes in Huge G: 403335\n",
      "Nb of edges in Huge G: 1211381\n"
     ]
    }
   ],
   "source": [
    "N_node_per_cluster = 50000\n",
    "self_loop_proba = 1/42000\n",
    "inter_cluster_edge_proba = 10/N_node_per_cluster\n",
    "inter_cluster_edge_bonus_proba = 2/N_node_per_cluster\n",
    "max_edges_between_clusters = 1000\n",
    "\n",
    "G = nx.DiGraph()\n",
    "G.add_edges_from([\n",
    "    ('A', 'A'),\n",
    "    ('A', 'C'),\n",
    "    ('A', 'B'),\n",
    "    ('A', 'D'),\n",
    "    ('A', 'X1'),\n",
    "    ('D', 'D'),\n",
    "    ('D', 'Y'),\n",
    "    ('D', 'X2'),\n",
    "    ('D', 'F'),\n",
    "    ('D', 'E'),\n",
    "    ('F', 'F'),\n",
    "    ('F', 'E'),\n",
    "    ('X2', 'C'),\n",
    "    ('X2', 'X1'),\n",
    "    ('X2', 'X2'),\n",
    "    ('Y', 'X3'),\n",
    "    ('X3', 'E'),\n",
    "    ('X1', 'B'),\n",
    "    ('X1', 'Y'),\n",
    "    ('Y', 'Y'),\n",
    "    ('Y', 'E'),\n",
    "])\n",
    "\n",
    "#random.seed(42)\n",
    "\n",
    "\n",
    "def find_name(node, i):\n",
    "    if node in ['X1','X2', 'X3','Y'] and i ==0:\n",
    "        return(node)\n",
    "    else:\n",
    "        return(node +str(i))\n",
    "\n",
    "\n",
    "Huge_G = nx.DiGraph()\n",
    "\n",
    "\n",
    "for node in G.nodes:\n",
    "    if node == 'X2' or node == \"X1\":\n",
    "        continue\n",
    "    \n",
    "    # Effective probability of an edge, considering bonus for self-loop\n",
    "    effective_proba = inter_cluster_edge_proba + ((node, node) in G.edges) * inter_cluster_edge_bonus_proba\n",
    "    \n",
    "    edges = []\n",
    "    for i in range(N_node_per_cluster):\n",
    "        n_edges = random.randint(1, 5)\n",
    "        edges.extend((find_name(node, i), find_name(node, str(random.randint(0, N_node_per_cluster)))) for _ in range(n_edges))\n",
    "\n",
    "    # Add the sampled edges to the graph\n",
    "    Huge_G.add_edges_from(edges)\n",
    "\n",
    "    # Step 3: Add inter-cluster edges based on G's edges\n",
    "print(\"Intra cluster edges done\")\n",
    "\n",
    "for u, v in G.edges:\n",
    "    if u == 'X2' and v == \"X1\":\n",
    "        continue\n",
    "    num_inter_edges = random.randint(1, max_edges_between_clusters)  # Random number of edges between clusters\n",
    "    for _ in range(num_inter_edges):\n",
    "        i = random.randint(0,N_node_per_cluster)\n",
    "        j = random.randint(0,N_node_per_cluster)\n",
    "        Huge_G.add_edge(find_name(u,i), find_name(v, j))  # Add an edge between clusters\n",
    "print(\"Inter cluster edges done\")\n",
    "    \n",
    "for node in Huge_G.nodes:\n",
    "    if random.random() < self_loop_proba:\n",
    "        Huge_G.add_edge(node, node)\n",
    "print(\"self loop done\")\n",
    "\n",
    "\n",
    "Huge_G.add_edge('X2', 'X3')\n",
    "Huge_G.add_edge('X1', 'Y')\n",
    "print(\"done!\")\n",
    "\n",
    "\n",
    "print(\"Nb of nodes in Huge G:\", Huge_G.number_of_nodes())\n",
    "print(\"Nb of edges in Huge G:\", Huge_G.number_of_edges())\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Interventions: [('X1', 't - 4'), ('X2', 't - 3'), ('X3', 't - 42')]\n",
      "True\n"
     ]
    }
   ],
   "source": [
    "Interventions = [('X1', 4), ('X2', 3), ('X3', 42)]\n",
    "print(\"Interventions:\", [(X,  f\"t - {gamma_x}\") for X, gamma_x in Interventions])\n",
    "ans = fast_is_ICA(Huge_G,Interventions)\n",
    "\n",
    "print(ans)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Experiments"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "10\n",
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n",
      "10\n",
      "11\n",
      "12\n",
      "13\n",
      "14\n",
      "15\n",
      "16\n",
      "17\n",
      "18\n",
      "19\n",
      "100\n",
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n",
      "10\n",
      "11\n",
      "12\n",
      "13\n",
      "14\n",
      "15\n",
      "16\n",
      "17\n",
      "18\n",
      "19\n",
      "1000\n",
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n",
      "10\n",
      "11\n",
      "12\n",
      "13\n",
      "14\n",
      "15\n",
      "16\n",
      "17\n",
      "18\n",
      "19\n",
      "10000\n",
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n",
      "10\n",
      "11\n",
      "12\n",
      "13\n",
      "14\n",
      "15\n",
      "16\n",
      "17\n",
      "18\n",
      "19\n",
      "100000\n",
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n",
      "10\n",
      "11\n",
      "12\n",
      "13\n",
      "14\n",
      "15\n",
      "16\n",
      "17\n",
      "18\n",
      "19\n"
     ]
    }
   ],
   "source": [
    "measure_exec_times = True\n",
    "store_IBC = True\n",
    "graph_node_sizes = [10, 100, 1000, 10000, 100000]\n",
    "edge_probabilities = [3/ size for size in graph_node_sizes]\n",
    "N_graphs_per_exp = 20\n",
    "num_interventions = 5\n",
    "intervention_on_y_probability = .2\n",
    "timeit_number = 5\n",
    "\n",
    "execution_time_means = []\n",
    "execution_time_stds = []\n",
    "ICB_graphs_ratio = []\n",
    "\n",
    "for num_nodes, edge_probability in zip(graph_node_sizes, edge_probabilities):\n",
    "    print(num_nodes)\n",
    "    exec_times = []\n",
    "    count_ICB = 0\n",
    "    for i in range(N_graphs_per_exp):\n",
    "        print(i)\n",
    "        G = generate_random_graph(num_nodes, edge_probability)\n",
    "        Interventions = generate_random_interventions(G, num_interventions, intervention_on_y_probability)\n",
    "        if measure_exec_times:\n",
    "            exec_time = timeit.timeit(lambda: fast_is_ICA(G, Interventions), number=timeit_number)/timeit_number\n",
    "            exec_times.append(exec_time)\n",
    "        if store_IBC:\n",
    "            count_ICB += fast_is_ICA(G, Interventions)\n",
    "    if measure_exec_times:\n",
    "        execution_time_means.append(np.mean(exec_times))\n",
    "        execution_time_stds.append(np.std(exec_times, ddof=1))\n",
    "    if store_IBC:\n",
    "        ICB_graphs_ratio.append(count_ICB/N_graphs_per_exp *100)\n",
    "    \n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "graph_node_sizes [10, 100, 1000, 10000, 100000]\n",
      "execution_time_means [0.00010309400153346361, 0.0004521230061072857, 0.00343131399829872, 0.05267661800025962, 1.0698565710010008]\n",
      "execution_time_stds [2.8079863194731723e-05, 7.241745061746056e-05, 0.0013111295564977915, 0.01784754474939452, 0.20911229325081465]\n",
      "ICA_graphs_ratio [20.0, 0.0, 10.0, 10.0, 0.0]\n"
     ]
    }
   ],
   "source": [
    "print('graph_node_sizes', graph_node_sizes)\n",
    "print('execution_time_means', execution_time_means)\n",
    "print('execution_time_stds', execution_time_stds)\n",
    "print('ICA_graphs_ratio', ICB_graphs_ratio)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Plot the graph with error bars\n",
    "plt.errorbar(\n",
    "    graph_node_sizes,\n",
    "    execution_time_means,\n",
    "    yerr=execution_time_stds,\n",
    "    fmt='-o',\n",
    "    capsize=5,\n",
    "    color='teal',\n",
    "    ecolor='coral',\n",
    "    elinewidth=1.5,\n",
    "    markerfacecolor='teal',\n",
    "    #label=\"Execution Time\"\n",
    ")\n",
    "\n",
    "# Set logarithmic scale for both axes\n",
    "plt.xscale('log')  # Logarithmic scale for the x-axis\n",
    "plt.yscale('log')  # Logarithmic scale for the y-axis\n",
    "\n",
    "plt.xticks(fontsize=12)\n",
    "plt.yticks(fontsize=12)\n",
    "\n",
    "# Add labels and a title to make the graph informative\n",
    "plt.xlabel(\"Graph Size (Number of Nodes)\", fontsize = 14)  # Label for the X-axis\n",
    "plt.ylabel(\"Mean Execution Time (seconds)\", fontsize = 14)  # Label for the Y-axis\n",
    "\n",
    "# Add grid lines for better readability\n",
    "plt.grid(True, linestyle='--', alpha=0.7)\n",
    "\n",
    "# Add a legend to identify the data\n",
    "#plt.legend()\n",
    "\n",
    "plt.savefig('execution_times.pdf', dpi=3000, format='pdf')\n",
    "\n",
    "# Display the graph\n",
    "plt.show()\n",
    "\n"
   ]
  }
 ],
 "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.11.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
