{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# A Graph Modelling Class \n",
    "We first introduce the modelling class to model dynamical systems graphs with additional allowing for intervention to model control-terms. The basic parts `NodeType`, `Endpoint`, `Node` and `Edge` only slightly changed (and slimmed down) from code taken from [causal-learn](https://github.com/py-why/causal-learn/tree/main), the graph class `GeneralGraph` entire new to model cyclic graphs with a dictionary-storage and the entire utility-methods also had to be changed. <br>\n",
    "The approach to find strongly connected components is adapted from [Find SCCs](https://www.geeksforgeeks.org/strongly-connected-components/#brute-force-approach-for-finding-strongly-connected-components) <br>"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from graph_modelling.Utils.GraphUtils import generate_DMG, calculate_E_separation_triple\n",
    "\n",
    "from graph_modelling.Utils.Utils import generate_directed_matrices, generate_symmetric_matrices, parallel_process_with_joblib, parallel_group_configurations, find_maximal_matrix_pair\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "print_all_configurations = True\n",
    "verbose = True\n",
    "\n",
    "n = 4 # 4\n",
    "num_workers = 64 #  64\n",
    "configurations = []  # Store all configurations\n",
    "global_counting_variable = 0  # Define a global counting variable\n",
    "\n",
    "\n",
    "\n",
    "directed_matrices = list(generate_directed_matrices(n))\n",
    "symmetric_matrices = list(generate_symmetric_matrices(n))\n",
    "\n",
    "configurations = parallel_process_with_joblib(directed_matrices, symmetric_matrices, num_workers=num_workers)\n",
    "\n",
    "print(f\"Total number of configurations: {len(configurations)}\")\n",
    "# Parallelize the grouping process\n",
    "grouped_configurations = parallel_group_configurations(configurations, num_workers=num_workers)\n",
    "\n",
    "\n",
    "for triple_key, configs in grouped_configurations.items(): # 3. Display grouped results and find maximal matrix pairs\n",
    "    if verbose == True:\n",
    "        print(f\"Independence triples: {triple_key}\")\n",
    "        print(f\"Configurations with these triples ({len(configs)} configurations):\")\n",
    "        if print_all_configurations == True:\n",
    "            for config in configs:\n",
    "                print(f\"Directed matrix:\\n{config['directed_matrix']}\")\n",
    "                print(f\"Symmetric matrix:\\n{config['symmetric_matrix']}\")\n",
    "\n",
    "    # Find the maximal matrix pair for this group\n",
    "    maximal_pair = find_maximal_matrix_pair(configs)\n",
    "    if maximal_pair is not None:\n",
    "        if verbose == True:\n",
    "            print(f\"Maximal matrix pair found:\\nDirected matrix:\\n{maximal_pair[0]}\\nSymmetric matrix:\\n{maximal_pair[1]}\")\n",
    "    else:\n",
    "        print(\"No maximal matrix pair found\")\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "jax_env",
   "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.10.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
