{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 157,
   "id": "6dfa4043",
   "metadata": {},
   "outputs": [],
   "source": [
    "import ast\n",
    "import json\n",
    "from pathlib import Path\n",
    "\n",
    "import networkx as nx\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 158,
   "id": "9a96a83b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<networkx.classes.digraph.DiGraph at 0x7c69e17d5130>"
      ]
     },
     "execution_count": 158,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "graphs = Path('eval/graphs').iterdir()\n",
    "\n",
    "G = nx.read_gexf(next(g for g in graphs if g.suffix == '.gexf'))\n",
    "G"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 159,
   "id": "1eb369d8",
   "metadata": {},
   "outputs": [],
   "source": [
    "tools = {}\n",
    "with open('frankenstein/tools/tool_schema.jsonl') as f:\n",
    "    for line in f:\n",
    "        obj = json.loads(line)\n",
    "        fn = obj['function']\n",
    "        name = fn['name']\n",
    "        params = fn['parameters']\n",
    "        required = set(params.get('required', []))\n",
    "        properties = params.get('properties', {})\n",
    "        types = {k: v.get('type', 'string') for k, v in properties.items()}\n",
    "        tools[name] = {'required': required, 'types': types}\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 160,
   "id": "dae2d0d4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Node _root has attributes {'type': 'question_param', 'call_index': 0, 'slot_property': 'SP.DYN.LE00.FE.IN', 'slot_subject': 'SAU', 'slot_region': 'Western Asia', 'slot_year': '2018', 'slot_property_original': 'Life expectancy at birth, female (years)', 'slot_subject_name': 'Saudi Arabia', 'label': 'What proportion of the total expected lifespan of females at birth was contributed by Saudi Arabia for the countries in Western Asia in 2018?'}.\n",
      "Node 0b4c6 has attributes {'call_index': 1, 'result': 'SP.DYN.LE00.FE.IN', 'arg_indicator_name': 'Life expectancy at birth, female (years)', 'label': 'get_indicator_code_from_name'}.\n",
      "Node df9a0 has attributes {'call_index': 2, 'result': 'SAU', 'arg_country_name': 'Saudi Arabia', 'label': 'get_country_code_from_name'}.\n",
      "Node d3f91 has attributes {'call_index': 3, 'result': '80.543', 'arg_country_code': 'SAU', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 879f7 has attributes {'call_index': 4, 'result': '[\"ARE\", \"ARM\", \"AZE\", \"BHR\", \"CYP\", \"GEO\", \"IRQ\", \"ISR\", \"JOR\", \"KWT\", \"LBN\", \"OMN\", \"PSE\", \"QAT\", \"SAU\", \"SYR\", \"TUR\", \"YEM\"]', 'arg_region': 'Western Asia', 'label': 'get_country_codes_in_region'}.\n",
      "Node 08aef has attributes {'call_index': 5, 'result': '83.912', 'arg_country_code': 'ARE', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 434be has attributes {'call_index': 6, 'result': '79.0', 'arg_country_code': 'ARM', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 579f1 has attributes {'call_index': 7, 'result': '75.946', 'arg_country_code': 'AZE', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node f0ac9 has attributes {'call_index': 8, 'result': '81.507', 'arg_country_code': 'BHR', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 47988 has attributes {'call_index': 9, 'result': '83.765', 'arg_country_code': 'CYP', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node f864c has attributes {'call_index': 10, 'result': '78.75', 'arg_country_code': 'GEO', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 5acb2 has attributes {'call_index': 11, 'result': '72.842', 'arg_country_code': 'IRQ', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 53f87 has attributes {'call_index': 12, 'result': '84.8', 'arg_country_code': 'ISR', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node d66f3 has attributes {'call_index': 13, 'result': '78.834', 'arg_country_code': 'JOR', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node a643c has attributes {'call_index': 14, 'result': '81.435', 'arg_country_code': 'KWT', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 8eb2a has attributes {'call_index': 15, 'result': '80.093', 'arg_country_code': 'LBN', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 96e05 has attributes {'call_index': 16, 'result': '81.793', 'arg_country_code': 'OMN', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node adbfa has attributes {'call_index': 17, 'result': '77.609', 'arg_country_code': 'PSE', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node af189 has attributes {'call_index': 18, 'result': '83.8', 'arg_country_code': 'QAT', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 47b25 has attributes {'call_index': 19, 'result': '72.185', 'arg_country_code': 'SYR', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node adc3f has attributes {'call_index': 20, 'result': '80.634', 'arg_country_code': 'TUR', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node 5e747 has attributes {'call_index': 21, 'result': '69.883', 'arg_country_code': 'YEM', 'arg_indicator_code': 'SP.DYN.LE00.FE.IN', 'arg_year': '2018', 'label': 'retrieve_value'}.\n",
      "Node c121b has attributes {'call_index': 22, 'result': '0.05643', 'arg_value_a': 80.543, 'arg_value_b': 1427.331, 'label': 'divide'}.\n",
      "⚠️ Node c121b (divide) is missing incoming edge for argument: value_b\n",
      "Node b706c has attributes {'call_index': 23, 'result': '0.05643', 'arg_answer': '0.05643', 'label': 'final_answer'}.\n"
     ]
    }
   ],
   "source": [
    "for node, data in G.nodes(data=True):\n",
    "    # data.pop('call_order', None)\n",
    "    # data.pop('id', None)\n",
    "    short_node_name = node[-5:]\n",
    "    print(f'Node {short_node_name} has attributes {data}.')\n",
    "    if 'label' in data:\n",
    "        action = data['label']\n",
    "        if action in tools:\n",
    "            tool_info = tools[action]\n",
    "            required = list(tool_info['required'])\n",
    "\n",
    "            node_tool_called = data.get('tool_called', None)\n",
    "            node_args_used = {k.removeprefix('arg_'): value for k, value in data.items() if k.startswith('arg_')}\n",
    "            node_tool_result = data.get('result', None)\n",
    "\n",
    "            # Gather all incoming edge labels for this node\n",
    "            incoming_edges = list(G.in_edges(node, data=True))\n",
    "            incoming_labels = set()\n",
    "            for src, dst, edge_data in incoming_edges:\n",
    "                edge_label = edge_data.get('label', None)\n",
    "                if edge_label and '=' in edge_label:\n",
    "                    arg_name = edge_label.split('=')[0]\n",
    "                    incoming_labels.add(arg_name)\n",
    "                elif edge_label:\n",
    "                    incoming_labels.add(edge_label)\n",
    "\n",
    "            # If any arg value is a stringified list/dict, recover it with ast.literal_eval\n",
    "\n",
    "            for arg, value in node_args_used.items():\n",
    "                if isinstance(value, str):\n",
    "                    try:\n",
    "                        # Only try to parse if it looks like a list/dict\n",
    "                        if (value.startswith('[') and value.endswith(']')) or (value.startswith('{') and value.endswith('}')):\n",
    "                            node_args_used[arg] = ast.literal_eval(value)\n",
    "                    except Exception:\n",
    "                        pass  # Leave as string if parsing fails\n",
    "\n",
    "            # Now check for each arg if there is an incoming edge with that label\n",
    "            for arg in node_args_used:\n",
    "                if arg not in incoming_labels:\n",
    "                    print(f'⚠️ Node {short_node_name} ({action}) is missing incoming edge for argument: {arg}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 161,
   "id": "728e1fd4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "# Graph Structure\n",
      "\n",
      "## Node `question_root`\n",
      "- **call_index:** `0`\n",
      "- **original question:** What proportion of the total expected lifespan of females at birth was contributed by Saudi Arabia for the countries in Western Asia in 2018?\n",
      "\n",
      "## Node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "- **call_index:** `1`\n",
      "- **tool_name:** `get_indicator_code_from_name`\n",
      "- **arguments:**\n",
      "    - `indicator_name` = `Life expectancy at birth, female (years)` _(from node `question_root`)_\n",
      "- **result:** `SP.DYN.LE00.FE.IN`\n",
      "\n",
      "## Node `chatcmpl-tool-cb28c312a2c241e3a4cb4ec6205df9a0`\n",
      "- **call_index:** `2`\n",
      "- **tool_name:** `get_country_code_from_name`\n",
      "- **arguments:**\n",
      "    - `country_name` = `Saudi Arabia` _(from node `question_root`)_\n",
      "- **result:** `SAU`\n",
      "\n",
      "## Node `chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91`\n",
      "- **call_index:** `3`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `SAU` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `80.543`\n",
      "\n",
      "## Node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "- **call_index:** `4`\n",
      "- **tool_name:** `get_country_codes_in_region`\n",
      "- **arguments:**\n",
      "    - `region` = `Western Asia` _(from node `question_root`)_\n",
      "- **result:** `[\"ARE\", \"ARM\", \"AZE\", \"BHR\", \"CYP\", \"GEO\", \"IRQ\", \"ISR\", \"JOR\", \"KWT\", \"LBN\", \"OMN\", \"PSE\", \"QAT\", \"SAU\", \"SYR\", \"TUR\", \"YEM\"]`\n",
      "\n",
      "## Node `chatcmpl-tool-c74132609ecb43708799e576ebe08aef`\n",
      "- **call_index:** `5`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `ARE` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `83.912`\n",
      "\n",
      "## Node `chatcmpl-tool-f1d3022310634ba3b4dcb6018eb434be`\n",
      "- **call_index:** `6`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `ARM` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `79.0`\n",
      "\n",
      "## Node `chatcmpl-tool-1a809012f153463889c2766933f579f1`\n",
      "- **call_index:** `7`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `AZE` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `75.946`\n",
      "\n",
      "## Node `chatcmpl-tool-6e44c4aef73442c2ac4c1452605f0ac9`\n",
      "- **call_index:** `8`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `BHR` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `81.507`\n",
      "\n",
      "## Node `chatcmpl-tool-2f6f3ad93dc740c89d020a5e32947988`\n",
      "- **call_index:** `9`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `CYP` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `83.765`\n",
      "\n",
      "## Node `chatcmpl-tool-bb2337cee589460cae87d77ee10f864c`\n",
      "- **call_index:** `10`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `GEO` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `78.75`\n",
      "\n",
      "## Node `chatcmpl-tool-70399fbb7858434ab2a2f769bda5acb2`\n",
      "- **call_index:** `11`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `IRQ` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `72.842`\n",
      "\n",
      "## Node `chatcmpl-tool-fe125e2cbf7a4f1eafbf044c56f53f87`\n",
      "- **call_index:** `12`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `ISR` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `84.8`\n",
      "\n",
      "## Node `chatcmpl-tool-9924382030c645da8be1f51f075d66f3`\n",
      "- **call_index:** `13`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `JOR` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `78.834`\n",
      "\n",
      "## Node `chatcmpl-tool-8e5292d61988474e99fa1094580a643c`\n",
      "- **call_index:** `14`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `KWT` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `81.435`\n",
      "\n",
      "## Node `chatcmpl-tool-26cf6c7c4bd74c6fb33c5c18a2a8eb2a`\n",
      "- **call_index:** `15`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `LBN` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `80.093`\n",
      "\n",
      "## Node `chatcmpl-tool-302a1fbcb1b6452f913968fd6d196e05`\n",
      "- **call_index:** `16`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `OMN` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `81.793`\n",
      "\n",
      "## Node `chatcmpl-tool-46b4755edc39460fabebe51bcc0adbfa`\n",
      "- **call_index:** `17`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `PSE` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `77.609`\n",
      "\n",
      "## Node `chatcmpl-tool-01f49cbc1815445484c77a25b5aaf189`\n",
      "- **call_index:** `18`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `QAT` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `83.8`\n",
      "\n",
      "## Node `chatcmpl-tool-e7a6e4b1b88b4c38963a002339547b25`\n",
      "- **call_index:** `19`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `SYR` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `72.185`\n",
      "\n",
      "## Node `chatcmpl-tool-f3b8d0bef65e4aa98962b6725bbadc3f`\n",
      "- **call_index:** `20`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `TUR` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `80.634`\n",
      "\n",
      "## Node `chatcmpl-tool-828eaf1080544997a22f10babab5e747`\n",
      "- **call_index:** `21`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `YEM` _(from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`)_\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` _(from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`)_\n",
      "    - `year` = `2018` _(from node `question_root`)_\n",
      "- **result:** `69.883`\n",
      "\n",
      "## Node `chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b`\n",
      "- **call_index:** `22`\n",
      "- **tool_name:** `divide`\n",
      "- **arguments:**\n",
      "    - `value_a` = `80.543` _(from node `chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91`)_\n",
      "    - `value_b` = `1427.331` ⚠️ **No incoming edge for `value_b`: not derived from a previous tool call.**\n",
      "- **result:** `0.05643`\n",
      "\n",
      "## Node `chatcmpl-tool-cc6e50b4f62d4706b27a7e32615b706c`\n",
      "- **call_index:** `23`\n",
      "- **tool_name:** `final_answer`\n",
      "- **arguments:**\n",
      "    - `answer` = `0.05643` _(from node `chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b`)_\n",
      "- **result:** `0.05643`\n",
      "\n"
     ]
    }
   ],
   "source": [
    "def print_graph_markdown_incoming(G, tools):\n",
    "    lines = ['# Graph Structure', '']\n",
    "\n",
    "    # Build a mapping from (node, arg_name) -> source node\n",
    "    arg_sources = {}\n",
    "    for src, dst, edata in G.edges(data=True):\n",
    "        slot = edata.get('label', None)\n",
    "        if slot:\n",
    "            # If label is like \"arg_name=val\", extract just arg_name\n",
    "            if '=' in slot:\n",
    "                arg = slot.split('=')[0]\n",
    "            else:\n",
    "                arg = slot\n",
    "            arg_sources[(dst, arg)] = src\n",
    "\n",
    "    for node, data in G.nodes(data=True):\n",
    "        lines.append(f'## Node `{node}`')\n",
    "        if data:\n",
    "            call_index = data.get('call_index', None)\n",
    "            tool_name = data.get('label', None)\n",
    "            result = data.get('result', None)\n",
    "            tool_args = {k.removeprefix('arg_'): v for k, v in data.items() if k.startswith('arg_')}\n",
    "\n",
    "            # Header\n",
    "            if call_index is not None:\n",
    "                lines.append(f'- **call_index:** `{call_index}`')\n",
    "            if tool_name is not None:\n",
    "                if node == 'question_root':\n",
    "                    lines.append(f'- **original question:** {tool_name}')\n",
    "                else:\n",
    "                    lines.append(f'- **tool_name:** `{tool_name}`')\n",
    "\n",
    "            # Arguments\n",
    "            if tool_args:\n",
    "                lines.append('- **arguments:**')\n",
    "                for arg, v in tool_args.items():\n",
    "                    origin = arg_sources.get((node, arg))\n",
    "                    if origin:\n",
    "                        lines.append(f'    - `{arg}` = `{v}` _(from node `{origin}`)_')\n",
    "                    else:\n",
    "                        lines.append(\n",
    "                            f'    - `{arg}` = `{v}` ⚠️ **No incoming edge for `{arg}`: not derived from a previous tool call.**'\n",
    "                        )\n",
    "\n",
    "            # Result\n",
    "            if result is not None:\n",
    "                lines.append(f'- **result:** `{result}`')\n",
    "        else:\n",
    "            lines.append('This node has no attributes.')\n",
    "        lines.append('')\n",
    "    print('\\n'.join(lines))\n",
    "\n",
    "\n",
    "print_graph_markdown_incoming(G, tools)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 169,
   "id": "fdefc71e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "# Graph Structure\n",
      "\n",
      "## Node List\n",
      "\n",
      "### Node `question_root`\n",
      "- **call_index:** `0`\n",
      "- **original question:** What proportion of the total expected lifespan of females at birth was contributed by Saudi Arabia for the countries in Western Asia in 2018?\n",
      "\n",
      "### Node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "- **call_index:** `1`\n",
      "- **tool_name:** `get_indicator_code_from_name`\n",
      "- **arguments:**\n",
      "    - `indicator_name` = `Life expectancy at birth, female (years)` from node `question_root`\n",
      "- **result:** `SP.DYN.LE00.FE.IN`\n",
      "\n",
      "### Node `chatcmpl-tool-cb28c312a2c241e3a4cb4ec6205df9a0`\n",
      "- **call_index:** `2`\n",
      "- **tool_name:** `get_country_code_from_name`\n",
      "- **arguments:**\n",
      "    - `country_name` = `Saudi Arabia` from node `question_root`\n",
      "- **result:** `SAU`\n",
      "\n",
      "### Node `chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91`\n",
      "- **call_index:** `3`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `SAU` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `80.543`\n",
      "\n",
      "### Node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "- **call_index:** `4`\n",
      "- **tool_name:** `get_country_codes_in_region`\n",
      "- **arguments:**\n",
      "    - `region` = `Western Asia` from node `question_root`\n",
      "- **result:** `[\"ARE\", \"ARM\", \"AZE\", \"BHR\", \"CYP\", \"GEO\", \"IRQ\", \"ISR\", \"JOR\", \"KWT\", \"LBN\", \"OMN\", \"PSE\", \"QAT\", \"SAU\", \"SYR\", \"TUR\", \"YEM\"]`\n",
      "\n",
      "### Node `chatcmpl-tool-c74132609ecb43708799e576ebe08aef`\n",
      "- **call_index:** `5`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `ARE` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `83.912`\n",
      "\n",
      "### Node `chatcmpl-tool-f1d3022310634ba3b4dcb6018eb434be`\n",
      "- **call_index:** `6`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `ARM` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `79.0`\n",
      "\n",
      "### Node `chatcmpl-tool-1a809012f153463889c2766933f579f1`\n",
      "- **call_index:** `7`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `AZE` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `75.946`\n",
      "\n",
      "### Node `chatcmpl-tool-6e44c4aef73442c2ac4c1452605f0ac9`\n",
      "- **call_index:** `8`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `BHR` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `81.507`\n",
      "\n",
      "### Node `chatcmpl-tool-2f6f3ad93dc740c89d020a5e32947988`\n",
      "- **call_index:** `9`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `CYP` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `83.765`\n",
      "\n",
      "### Node `chatcmpl-tool-bb2337cee589460cae87d77ee10f864c`\n",
      "- **call_index:** `10`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `GEO` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `78.75`\n",
      "\n",
      "### Node `chatcmpl-tool-70399fbb7858434ab2a2f769bda5acb2`\n",
      "- **call_index:** `11`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `IRQ` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `72.842`\n",
      "\n",
      "### Node `chatcmpl-tool-fe125e2cbf7a4f1eafbf044c56f53f87`\n",
      "- **call_index:** `12`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `ISR` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `84.8`\n",
      "\n",
      "### Node `chatcmpl-tool-9924382030c645da8be1f51f075d66f3`\n",
      "- **call_index:** `13`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `JOR` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `78.834`\n",
      "\n",
      "### Node `chatcmpl-tool-8e5292d61988474e99fa1094580a643c`\n",
      "- **call_index:** `14`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `KWT` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `81.435`\n",
      "\n",
      "### Node `chatcmpl-tool-26cf6c7c4bd74c6fb33c5c18a2a8eb2a`\n",
      "- **call_index:** `15`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `LBN` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `80.093`\n",
      "\n",
      "### Node `chatcmpl-tool-302a1fbcb1b6452f913968fd6d196e05`\n",
      "- **call_index:** `16`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `OMN` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `81.793`\n",
      "\n",
      "### Node `chatcmpl-tool-46b4755edc39460fabebe51bcc0adbfa`\n",
      "- **call_index:** `17`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `PSE` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `77.609`\n",
      "\n",
      "### Node `chatcmpl-tool-01f49cbc1815445484c77a25b5aaf189`\n",
      "- **call_index:** `18`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `QAT` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `83.8`\n",
      "\n",
      "### Node `chatcmpl-tool-e7a6e4b1b88b4c38963a002339547b25`\n",
      "- **call_index:** `19`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `SYR` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `72.185`\n",
      "\n",
      "### Node `chatcmpl-tool-f3b8d0bef65e4aa98962b6725bbadc3f`\n",
      "- **call_index:** `20`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `TUR` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `80.634`\n",
      "\n",
      "### Node `chatcmpl-tool-828eaf1080544997a22f10babab5e747`\n",
      "- **call_index:** `21`\n",
      "- **tool_name:** `retrieve_value`\n",
      "- **arguments:**\n",
      "    - `country_code` = `YEM` from node `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7`\n",
      "    - `indicator_code` = `SP.DYN.LE00.FE.IN` from node `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6`\n",
      "    - `year` = `2018` from node `question_root`\n",
      "- **result:** `69.883`\n",
      "\n",
      "### Node `chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b`\n",
      "- **call_index:** `22`\n",
      "- **tool_name:** `divide`\n",
      "- **arguments:**\n",
      "    - `value_a` = `80.543` from node `chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91`\n",
      "    - `value_b` = `1427.331` ⚠️ **No incoming edge. This argument is not derived from any previous node, so its provenance is unclear.**\n",
      "- **result:** `0.05643`\n",
      "\n",
      "### Node `chatcmpl-tool-cc6e50b4f62d4706b27a7e32615b706c`\n",
      "- **call_index:** `23`\n",
      "- **tool_name:** `final_answer`\n",
      "- **arguments:**\n",
      "    - `answer` = `0.05643` from node `chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b`\n",
      "- **result:** `0.05643`\n",
      "\n",
      "---\n",
      "## Edges\n",
      "- `question_root` → `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` (arg: `indicator_name=\"Life expectancy at birth, female (years)\"`)\n",
      "- `question_root` → `chatcmpl-tool-cb28c312a2c241e3a4cb4ec6205df9a0` (arg: `country_name=Saudi Arabia`)\n",
      "- `question_root` → `chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` (arg: `region=Western Asia`)\n",
      "- `question_root` → `chatcmpl-tool-c74132609ecb43708799e576ebe08aef` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-f1d3022310634ba3b4dcb6018eb434be` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-1a809012f153463889c2766933f579f1` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-6e44c4aef73442c2ac4c1452605f0ac9` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-2f6f3ad93dc740c89d020a5e32947988` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-bb2337cee589460cae87d77ee10f864c` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-70399fbb7858434ab2a2f769bda5acb2` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-fe125e2cbf7a4f1eafbf044c56f53f87` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-9924382030c645da8be1f51f075d66f3` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-8e5292d61988474e99fa1094580a643c` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-26cf6c7c4bd74c6fb33c5c18a2a8eb2a` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-302a1fbcb1b6452f913968fd6d196e05` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-46b4755edc39460fabebe51bcc0adbfa` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-01f49cbc1815445484c77a25b5aaf189` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-e7a6e4b1b88b4c38963a002339547b25` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-f3b8d0bef65e4aa98962b6725bbadc3f` (arg: `year=2018`)\n",
      "- `question_root` → `chatcmpl-tool-828eaf1080544997a22f10babab5e747` (arg: `year=2018`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-c74132609ecb43708799e576ebe08aef` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-f1d3022310634ba3b4dcb6018eb434be` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-1a809012f153463889c2766933f579f1` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-6e44c4aef73442c2ac4c1452605f0ac9` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-2f6f3ad93dc740c89d020a5e32947988` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-bb2337cee589460cae87d77ee10f864c` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-70399fbb7858434ab2a2f769bda5acb2` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-fe125e2cbf7a4f1eafbf044c56f53f87` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-9924382030c645da8be1f51f075d66f3` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-8e5292d61988474e99fa1094580a643c` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-26cf6c7c4bd74c6fb33c5c18a2a8eb2a` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-302a1fbcb1b6452f913968fd6d196e05` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-46b4755edc39460fabebe51bcc0adbfa` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-01f49cbc1815445484c77a25b5aaf189` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-e7a6e4b1b88b4c38963a002339547b25` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-f3b8d0bef65e4aa98962b6725bbadc3f` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6` → `chatcmpl-tool-828eaf1080544997a22f10babab5e747` (arg: `indicator_code=SP.DYN.LE00.FE.IN`)\n",
      "- `chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91` → `chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b` (arg: `value_a=80.543`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91` (arg: `country_code=SAU`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-c74132609ecb43708799e576ebe08aef` (arg: `country_code=ARE`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-f1d3022310634ba3b4dcb6018eb434be` (arg: `country_code=ARM`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-1a809012f153463889c2766933f579f1` (arg: `country_code=AZE`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-6e44c4aef73442c2ac4c1452605f0ac9` (arg: `country_code=BHR`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-2f6f3ad93dc740c89d020a5e32947988` (arg: `country_code=CYP`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-bb2337cee589460cae87d77ee10f864c` (arg: `country_code=GEO`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-70399fbb7858434ab2a2f769bda5acb2` (arg: `country_code=IRQ`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-fe125e2cbf7a4f1eafbf044c56f53f87` (arg: `country_code=ISR`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-9924382030c645da8be1f51f075d66f3` (arg: `country_code=JOR`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-8e5292d61988474e99fa1094580a643c` (arg: `country_code=KWT`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-26cf6c7c4bd74c6fb33c5c18a2a8eb2a` (arg: `country_code=LBN`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-302a1fbcb1b6452f913968fd6d196e05` (arg: `country_code=OMN`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-46b4755edc39460fabebe51bcc0adbfa` (arg: `country_code=PSE`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-01f49cbc1815445484c77a25b5aaf189` (arg: `country_code=QAT`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-e7a6e4b1b88b4c38963a002339547b25` (arg: `country_code=SYR`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-f3b8d0bef65e4aa98962b6725bbadc3f` (arg: `country_code=TUR`)\n",
      "- `chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7` → `chatcmpl-tool-828eaf1080544997a22f10babab5e747` (arg: `country_code=YEM`)\n",
      "- `chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b` → `chatcmpl-tool-cc6e50b4f62d4706b27a7e32615b706c` (arg: `answer=0.05643`)\n",
      "---\n",
      "## ⚠️ Issues\n",
      "- Node `chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b` arg `value_b` = `1427.331` has no incoming edge.\n"
     ]
    }
   ],
   "source": [
    "def print_graph_markdown_incoming(G, tools):\n",
    "    lines = ['# Graph Structure', '', '## Node List', '']\n",
    "\n",
    "    # Build a mapping from (node, arg_name) -> source node\n",
    "    arg_sources = {}\n",
    "    issues = []\n",
    "    for src, dst, edata in G.edges(data=True):\n",
    "        slot = edata.get('label', None)\n",
    "        if slot:\n",
    "            arg = slot.split('=')[0] if '=' in slot else slot\n",
    "            arg_sources[(dst, arg)] = src\n",
    "\n",
    "    # Node descriptions\n",
    "    for node, data in G.nodes(data=True):\n",
    "        lines.append(f'### Node `{node}`')\n",
    "        if data:\n",
    "            call_index = data.get('call_index')\n",
    "            tool_name = data.get('label')\n",
    "            result = data.get('result')\n",
    "            tool_args = {k.removeprefix('arg_'): v for k, v in data.items() if k.startswith('arg_')}\n",
    "\n",
    "            # Header\n",
    "            if call_index is not None:\n",
    "                lines.append(f'- **call_index:** `{call_index}`')\n",
    "            if tool_name is not None:\n",
    "                if node == 'question_root':\n",
    "                    lines.append(f'- **original question:** {tool_name}')\n",
    "                else:\n",
    "                    lines.append(f'- **tool_name:** `{tool_name}`')\n",
    "\n",
    "            # Arguments\n",
    "            if tool_args:\n",
    "                lines.append('- **arguments:**')\n",
    "                for arg, v in tool_args.items():\n",
    "                    origin = arg_sources.get((node, arg))\n",
    "                    if origin:\n",
    "                        lines.append(f'    - `{arg}` = `{v}` from node `{origin}`')\n",
    "                    else:\n",
    "                        warning = f'Node `{node}` arg `{arg}` = `{v}` has no incoming edge.'\n",
    "                        lines.append(\n",
    "                            f'    - `{arg}` = `{v}` ⚠️ **No incoming edge. This argument is not derived from any previous node, so its provenance is unclear.**'\n",
    "                        )\n",
    "                        issues.append(warning)\n",
    "\n",
    "            # Result\n",
    "            if result is not None:\n",
    "                lines.append(f'- **result:** `{result}`')\n",
    "        else:\n",
    "            lines.append('This node has no attributes.')\n",
    "        lines.append('')\n",
    "\n",
    "    # Explicit Edges section\n",
    "    lines.append('---')\n",
    "    lines.append('## Edges')\n",
    "    for src, dst, edata in G.edges(data=True):\n",
    "        label = edata.get('label', '')\n",
    "        if label:\n",
    "            lines.append(f'- `{src}` → `{dst}` (arg: `{label}`)')\n",
    "        else:\n",
    "            lines.append(f'- `{src}` → `{dst}`')\n",
    "\n",
    "    # Issues summary\n",
    "    if issues:\n",
    "        lines.append('---')\n",
    "        lines.append('## ⚠️ Issues')\n",
    "        for w in issues:\n",
    "            lines.append(f'- {w}')\n",
    "\n",
    "    print('\\n'.join(lines))\n",
    "\n",
    "\n",
    "print_graph_markdown_incoming(G, tools)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "eedd2728",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "nodes:\n",
      "- id: question_root\n",
      "  call_index: 0\n",
      "  tool_name: null\n",
      "  original_question: What proportion of the total expected lifespan of females at\n",
      "    birth was contributed by Saudi Arabia for the countries in Western Asia in 2018?\n",
      "- id: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  call_index: 1\n",
      "  tool_name: get_indicator_code_from_name\n",
      "  arguments:\n",
      "  - name: indicator_name\n",
      "    value: Life expectancy at birth, female (years)\n",
      "    source_node: question_root\n",
      "  result: SP.DYN.LE00.FE.IN\n",
      "- id: chatcmpl-tool-cb28c312a2c241e3a4cb4ec6205df9a0\n",
      "  call_index: 2\n",
      "  tool_name: get_country_code_from_name\n",
      "  arguments:\n",
      "  - name: country_name\n",
      "    value: Saudi Arabia\n",
      "    source_node: question_root\n",
      "  result: SAU\n",
      "- id: chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91\n",
      "  call_index: 3\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: SAU\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '80.543'\n",
      "- id: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  call_index: 4\n",
      "  tool_name: get_country_codes_in_region\n",
      "  arguments:\n",
      "  - name: region\n",
      "    value: Western Asia\n",
      "    source_node: question_root\n",
      "  result: '[\"ARE\", \"ARM\", \"AZE\", \"BHR\", \"CYP\", \"GEO\", \"IRQ\", \"ISR\", \"JOR\", \"KWT\",\n",
      "    \"LBN\", \"OMN\", \"PSE\", \"QAT\", \"SAU\", \"SYR\", \"TUR\", \"YEM\"]'\n",
      "- id: chatcmpl-tool-c74132609ecb43708799e576ebe08aef\n",
      "  call_index: 5\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: ARE\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '83.912'\n",
      "- id: chatcmpl-tool-f1d3022310634ba3b4dcb6018eb434be\n",
      "  call_index: 6\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: ARM\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '79.0'\n",
      "- id: chatcmpl-tool-1a809012f153463889c2766933f579f1\n",
      "  call_index: 7\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: AZE\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '75.946'\n",
      "- id: chatcmpl-tool-6e44c4aef73442c2ac4c1452605f0ac9\n",
      "  call_index: 8\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: BHR\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '81.507'\n",
      "- id: chatcmpl-tool-2f6f3ad93dc740c89d020a5e32947988\n",
      "  call_index: 9\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: CYP\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '83.765'\n",
      "- id: chatcmpl-tool-bb2337cee589460cae87d77ee10f864c\n",
      "  call_index: 10\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: GEO\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '78.75'\n",
      "- id: chatcmpl-tool-70399fbb7858434ab2a2f769bda5acb2\n",
      "  call_index: 11\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: IRQ\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '72.842'\n",
      "- id: chatcmpl-tool-fe125e2cbf7a4f1eafbf044c56f53f87\n",
      "  call_index: 12\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: ISR\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '84.8'\n",
      "- id: chatcmpl-tool-9924382030c645da8be1f51f075d66f3\n",
      "  call_index: 13\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: JOR\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '78.834'\n",
      "- id: chatcmpl-tool-8e5292d61988474e99fa1094580a643c\n",
      "  call_index: 14\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: KWT\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '81.435'\n",
      "- id: chatcmpl-tool-26cf6c7c4bd74c6fb33c5c18a2a8eb2a\n",
      "  call_index: 15\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: LBN\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '80.093'\n",
      "- id: chatcmpl-tool-302a1fbcb1b6452f913968fd6d196e05\n",
      "  call_index: 16\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: OMN\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '81.793'\n",
      "- id: chatcmpl-tool-46b4755edc39460fabebe51bcc0adbfa\n",
      "  call_index: 17\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: PSE\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '77.609'\n",
      "- id: chatcmpl-tool-01f49cbc1815445484c77a25b5aaf189\n",
      "  call_index: 18\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: QAT\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '83.8'\n",
      "- id: chatcmpl-tool-e7a6e4b1b88b4c38963a002339547b25\n",
      "  call_index: 19\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: SYR\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '72.185'\n",
      "- id: chatcmpl-tool-f3b8d0bef65e4aa98962b6725bbadc3f\n",
      "  call_index: 20\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: TUR\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '80.634'\n",
      "- id: chatcmpl-tool-828eaf1080544997a22f10babab5e747\n",
      "  call_index: 21\n",
      "  tool_name: retrieve_value\n",
      "  arguments:\n",
      "  - name: country_code\n",
      "    value: YEM\n",
      "    source_node: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  - name: indicator_code\n",
      "    value: SP.DYN.LE00.FE.IN\n",
      "    source_node: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  - name: year\n",
      "    value: '2018'\n",
      "    source_node: question_root\n",
      "  result: '69.883'\n",
      "- id: chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b\n",
      "  call_index: 22\n",
      "  tool_name: divide\n",
      "  arguments:\n",
      "  - name: value_a\n",
      "    value: 80.543\n",
      "    source_node: chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91\n",
      "  - name: value_b\n",
      "    value: 1427.331\n",
      "    source_node: null\n",
      "  result: '0.05643'\n",
      "- id: chatcmpl-tool-cc6e50b4f62d4706b27a7e32615b706c\n",
      "  call_index: 23\n",
      "  tool_name: final_answer\n",
      "  arguments:\n",
      "  - name: answer\n",
      "    value: '0.05643'\n",
      "    source_node: chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b\n",
      "  result: '0.05643'\n",
      "edges:\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  arg:\n",
      "    indicator_name: Life expectancy at birth, female (years)\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-cb28c312a2c241e3a4cb4ec6205df9a0\n",
      "  arg:\n",
      "    country_name: Saudi Arabia\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  arg:\n",
      "    region: Western Asia\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-c74132609ecb43708799e576ebe08aef\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-f1d3022310634ba3b4dcb6018eb434be\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-1a809012f153463889c2766933f579f1\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-6e44c4aef73442c2ac4c1452605f0ac9\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-2f6f3ad93dc740c89d020a5e32947988\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-bb2337cee589460cae87d77ee10f864c\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-70399fbb7858434ab2a2f769bda5acb2\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-fe125e2cbf7a4f1eafbf044c56f53f87\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-9924382030c645da8be1f51f075d66f3\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-8e5292d61988474e99fa1094580a643c\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-26cf6c7c4bd74c6fb33c5c18a2a8eb2a\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-302a1fbcb1b6452f913968fd6d196e05\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-46b4755edc39460fabebe51bcc0adbfa\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-01f49cbc1815445484c77a25b5aaf189\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-e7a6e4b1b88b4c38963a002339547b25\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-f3b8d0bef65e4aa98962b6725bbadc3f\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: question_root\n",
      "  to: chatcmpl-tool-828eaf1080544997a22f10babab5e747\n",
      "  arg:\n",
      "    year: '2018'\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-c74132609ecb43708799e576ebe08aef\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-f1d3022310634ba3b4dcb6018eb434be\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-1a809012f153463889c2766933f579f1\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-6e44c4aef73442c2ac4c1452605f0ac9\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-2f6f3ad93dc740c89d020a5e32947988\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-bb2337cee589460cae87d77ee10f864c\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-70399fbb7858434ab2a2f769bda5acb2\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-fe125e2cbf7a4f1eafbf044c56f53f87\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-9924382030c645da8be1f51f075d66f3\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-8e5292d61988474e99fa1094580a643c\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-26cf6c7c4bd74c6fb33c5c18a2a8eb2a\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-302a1fbcb1b6452f913968fd6d196e05\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-46b4755edc39460fabebe51bcc0adbfa\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-01f49cbc1815445484c77a25b5aaf189\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-e7a6e4b1b88b4c38963a002339547b25\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-f3b8d0bef65e4aa98962b6725bbadc3f\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-446b1ed010ba47ba828047e6fa30b4c6\n",
      "  to: chatcmpl-tool-828eaf1080544997a22f10babab5e747\n",
      "  arg:\n",
      "    indicator_code: SP.DYN.LE00.FE.IN\n",
      "- from: chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91\n",
      "  to: chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b\n",
      "  arg:\n",
      "    value_a: 80.543\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-013450c450734cc29d8a2f8ea2cd3f91\n",
      "  arg:\n",
      "    country_code: SAU\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-c74132609ecb43708799e576ebe08aef\n",
      "  arg:\n",
      "    country_code: ARE\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-f1d3022310634ba3b4dcb6018eb434be\n",
      "  arg:\n",
      "    country_code: ARM\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-1a809012f153463889c2766933f579f1\n",
      "  arg:\n",
      "    country_code: AZE\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-6e44c4aef73442c2ac4c1452605f0ac9\n",
      "  arg:\n",
      "    country_code: BHR\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-2f6f3ad93dc740c89d020a5e32947988\n",
      "  arg:\n",
      "    country_code: CYP\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-bb2337cee589460cae87d77ee10f864c\n",
      "  arg:\n",
      "    country_code: GEO\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-70399fbb7858434ab2a2f769bda5acb2\n",
      "  arg:\n",
      "    country_code: IRQ\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-fe125e2cbf7a4f1eafbf044c56f53f87\n",
      "  arg:\n",
      "    country_code: ISR\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-9924382030c645da8be1f51f075d66f3\n",
      "  arg:\n",
      "    country_code: JOR\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-8e5292d61988474e99fa1094580a643c\n",
      "  arg:\n",
      "    country_code: KWT\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-26cf6c7c4bd74c6fb33c5c18a2a8eb2a\n",
      "  arg:\n",
      "    country_code: LBN\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-302a1fbcb1b6452f913968fd6d196e05\n",
      "  arg:\n",
      "    country_code: OMN\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-46b4755edc39460fabebe51bcc0adbfa\n",
      "  arg:\n",
      "    country_code: PSE\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-01f49cbc1815445484c77a25b5aaf189\n",
      "  arg:\n",
      "    country_code: QAT\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-e7a6e4b1b88b4c38963a002339547b25\n",
      "  arg:\n",
      "    country_code: SYR\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-f3b8d0bef65e4aa98962b6725bbadc3f\n",
      "  arg:\n",
      "    country_code: TUR\n",
      "- from: chatcmpl-tool-e223a6e9e96a4a9fb6388b42457879f7\n",
      "  to: chatcmpl-tool-828eaf1080544997a22f10babab5e747\n",
      "  arg:\n",
      "    country_code: YEM\n",
      "- from: chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b\n",
      "  to: chatcmpl-tool-cc6e50b4f62d4706b27a7e32615b706c\n",
      "  arg:\n",
      "    answer: '0.05643'\n",
      "issues:\n",
      "- Node `chatcmpl-tool-028fdfd66a8145a39d0565e68d7c121b` arg `value_b` = `1427.331`\n",
      "  has no incoming edge, indicating that it is not derived from a previous tool call\n",
      "  and so its provenance is unclear.\n",
      "\n"
     ]
    }
   ],
   "source": [
    "import yaml\n",
    "\n",
    "\n",
    "def generate_graph_yaml(G, tools):\n",
    "    \"\"\"Generate a YAML-style report of a tool-use graph, with explicit argument provenance.\"\"\"\n",
    "    report = {}\n",
    "\n",
    "    # Build mapping from (node, arg_name) -> source node\n",
    "    arg_sources = {}\n",
    "    issues = []\n",
    "    for src, dst, edata in G.edges(data=True):\n",
    "        slot = edata.get('label', None)\n",
    "        if slot:\n",
    "            arg = slot.split('=')[0] if '=' in slot else slot\n",
    "            arg_sources[(dst, arg)] = src\n",
    "\n",
    "    # Nodes\n",
    "    nodes_list = []\n",
    "    for node, data in G.nodes(data=True):\n",
    "        node_dict = {'id': node}\n",
    "        if data:\n",
    "            call_index = data.get('call_index')\n",
    "            tool_name = data.get('label')\n",
    "            result = data.get('result')\n",
    "            tool_args = {k.removeprefix('arg_'): v for k, v in data.items() if k.startswith('arg_')}\n",
    "\n",
    "            if call_index is not None:\n",
    "                node_dict['call_index'] = call_index\n",
    "            if tool_name is not None:\n",
    "                node_dict['tool_name'] = tool_name if node != 'question_root' else None\n",
    "                if node == 'question_root':\n",
    "                    node_dict['original_question'] = tool_name\n",
    "\n",
    "            # Arguments with provenance\n",
    "            if tool_args:\n",
    "                args_list = []\n",
    "                for arg, v in tool_args.items():\n",
    "                    origin = arg_sources.get((node, arg))\n",
    "                    if not origin:\n",
    "                        issues.append(\n",
    "                            f'Node `{node}` arg `{arg}` = `{v}` has no incoming edge, indicating that it is not derived from a previous tool call and so its provenance is unclear.'\n",
    "                        )\n",
    "                        origin = None\n",
    "                    args_list.append({'name': arg, 'value': v, 'source_node': origin})\n",
    "                node_dict['arguments'] = args_list\n",
    "\n",
    "            # Result\n",
    "            if result is not None:\n",
    "                node_dict['result'] = result\n",
    "        else:\n",
    "            node_dict['note'] = 'No attributes'\n",
    "\n",
    "        nodes_list.append(node_dict)\n",
    "    report['nodes'] = nodes_list\n",
    "\n",
    "    # Edges\n",
    "    edges_list = []\n",
    "    for src, dst, edata in G.edges(data=True):\n",
    "        label = edata.get('label', None)\n",
    "        edge_dict = {'from': src, 'to': dst, 'arg': None}\n",
    "        if label:\n",
    "            arg, value = label.split('=')\n",
    "            edge_dict['arg'] = {arg: value}\n",
    "        edges_list.append(edge_dict)\n",
    "    report['edges'] = edges_list\n",
    "\n",
    "    # Issues\n",
    "    report['issues'] = issues\n",
    "\n",
    "    # Output YAML\n",
    "    yaml_str = yaml.dump(report, sort_keys=False, allow_unicode=True)\n",
    "    return yaml_str\n",
    "\n",
    "\n",
    "# Example usage:\n",
    "yaml_report = generate_graph_yaml(G, tools)\n",
    "with open('graph_report.yaml', 'w', encoding='utf-8') as f:\n",
    "    f.write(yaml_report)\n",
    "print(yaml_report)\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "frankenstein",
   "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.12.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
