{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "06544132",
   "metadata": {},
   "source": [
    "# Client demo for the Kimina Lean Server\n",
    "\n",
    "This notebook shows how to use the Kimina Lean Server for three different tasks:\n",
    "1. Large-scale verification: send a large batch of Lean codes to the server and get verification feedbacks.\n",
    "2. Proof data extraction: extract all tactics and tactic states from a Lean code.\n",
    "3. Interaction: send code, receive verification and extracted data, update the code, and repeat.\n",
    "\n",
    "For all three tasks, you need to launch the Kimina Lean Server first. You can do this by running the following command in your terminal:\n",
    "```bash\n",
    "python -m server\n",
    "```\n",
    "\n",
    "## 1. Large-scale verification\n",
    "\n",
    "This first section demonstrates how to use the Kimina Lean Server for large-scale verification.\n",
    "\n",
    "### 1.1. Import libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ddbea2f1",
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "import nest_asyncio\n",
    "import uuid\n",
    "\n",
    "sys.path.insert(0, \"YOUR_PATH_TO_kimina-lean-server\")\n",
    "\n",
    "from client.client import Lean4Client\n",
    "from client.infotree import extract_data\n",
    "from utils.proof_utils import split_proof_header, parse_client_response\n",
    "\n",
    "nest_asyncio.apply()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d89bc7c8",
   "metadata": {},
   "source": [
    "### 1.2. Initialize a client"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "8deb7546",
   "metadata": {},
   "outputs": [],
   "source": [
    "client = Lean4Client(base_url=\"http://127.0.0.1:12332\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "76e773d4",
   "metadata": {},
   "source": [
    "### 1.3. Load Lean codes\n",
    "\n",
    "Let's load Lean codes that we will verify. For this demo, we will use the Lean Workbook dataset."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2a2e2f70",
   "metadata": {},
   "outputs": [],
   "source": [
    "from datasets import load_dataset\n",
    "\n",
    "dataset = load_dataset(\"Goedel-LM/Lean-workbook-proofs\", split=\"train\")\n",
    "dataset = dataset.shuffle(seed=42)\n",
    "dataset = dataset.select(range(1000))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "2b3f3a1b",
   "metadata": {},
   "outputs": [],
   "source": [
    "codes = [\n",
    "    {\"custom_id\": sample[\"problem_id\"], \"proof\": sample[\"full_proof\"]}\n",
    "    for sample in dataset\n",
    "]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "91773822",
   "metadata": {},
   "source": [
    "### 1.4. Send codes to the server"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "f082aba0",
   "metadata": {},
   "outputs": [],
   "source": [
    "response = client.verify(codes, timeout=60)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "5b5ddc33",
   "metadata": {},
   "outputs": [],
   "source": [
    "verification_results = [\n",
    "    parse_client_response(item)\n",
    "    for item in response[\"results\"]\n",
    "]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b6c58645",
   "metadata": {},
   "source": [
    "We can now have access to the verification result of any proof. For example, here is the verification result of the first proof in the dataset."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "0b96dd0f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'has_error': False, 'is_valid_no_sorry': True, 'is_valid_with_sorry': True}\n"
     ]
    }
   ],
   "source": [
    "print(verification_results[0])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5570fc9e",
   "metadata": {},
   "source": [
    "As we can see, this proof doesn't have any errors, is valid and doesn't use `sorry`.\n",
    "\n",
    "Let's check the percentage of proofs that are valid without using `sorry`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "70b167bc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Valid rate: 97.50%\n"
     ]
    }
   ],
   "source": [
    "count_valid_no_sorry = sum(r[\"is_valid_no_sorry\"] for r in verification_results)\n",
    "\n",
    "valid_rate = count_valid_no_sorry / len(verification_results)\n",
    "print(f\"Valid rate: {valid_rate:.2%}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "41085638",
   "metadata": {},
   "source": [
    "Not 100% of the proofs are valid, for two reasons: \n",
    "1. These proofs were originally written with Lean 4.9.0, and our server is using Lean 4.15.0.\n",
    "2. We are using a timeout of 60 seconds for the verification. The timeout can be set to a higher value if needed."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d28f7e16",
   "metadata": {},
   "source": [
    "## 2. Data extraction\n",
    "\n",
    "This second section demonstrates how to use the Kimina Lean Server to extract proof data (tactics and tactic states) from a Lean code.\n",
    "\n",
    "### 2.1. Import libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2932a6ca",
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "import nest_asyncio\n",
    "import uuid\n",
    "\n",
    "sys.path.insert(0, \"YOUR_PATH_TO_kimina-lean-server\")\n",
    "\n",
    "from client.client import Lean4Client\n",
    "from client.infotree import extract_data\n",
    "from utils.proof_utils import split_proof_header\n",
    "\n",
    "nest_asyncio.apply()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4bae4094",
   "metadata": {},
   "source": [
    "### 2.2. Initialize a client"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "1504c561",
   "metadata": {},
   "outputs": [],
   "source": [
    "client = Lean4Client(base_url=\"http://127.0.0.1:12332\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6db81518",
   "metadata": {},
   "source": [
    "### 2.3. Load the Lean code\n",
    "\n",
    "Let's load the Lean code from a file. The code should contain the imports, a statement and its proof."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "3efe3005",
   "metadata": {},
   "outputs": [],
   "source": [
    "lean_file = \"demo_extraction.lean\"\n",
    "with open(lean_file, \"r\") as f:\n",
    "    lean_code = f.read()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "08624aac",
   "metadata": {},
   "source": [
    "For this demo, we will extract tactics and tactic states from the following proof from Lean Workbook."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "e41498b4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "import Mathlib\n",
      "import Aesop\n",
      "\n",
      "set_option maxHeartbeats 0\n",
      "\n",
      "open BigOperators Real Nat Topology Rat\n",
      "\n",
      "/- Let $a,b,c>0$ . Prove that: $\\frac{a^{2}}{b+c}+\\frac{b^{2}}{a+c}+\\frac{16c^{2}}{a+b}\\geq \\frac{1}{9}(64c-a-b)$ -/\n",
      "theorem lean_workbook_10058 (a b c : ℝ) (ha : 0 < a) (hb : 0 < b) (hc : 0 < c) : (a^2 / (b + c) + b^2 / (a + c) + 16 * c^2 / (a + b)) ≥ (1 / 9) * (64 * c - a - b)  := by\n",
      "  have h₁ : 0 < a + b + c := by linarith\n",
      "  have h₂ : 0 < a * b := by positivity\n",
      "  have h₃ : 0 < a * c := by positivity\n",
      "  have h₄ : 0 < b * c := by positivity\n",
      "  field_simp [h₁.ne', h₂.ne', h₃.ne', h₄.ne']\n",
      "  rw [div_le_div_iff]\n",
      "  nlinarith [sq_nonneg (a - b), sq_nonneg (a - 4 * c), sq_nonneg (b - 4 * c), sq_nonneg (a + b - 4 * c),\n",
      "    sq_nonneg (a + b - 2 * c), sq_nonneg (a + b + 2 * c)]\n",
      "  all_goals nlinarith\n"
     ]
    }
   ],
   "source": [
    "print(lean_code)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "73c67896",
   "metadata": {},
   "source": [
    "### 2.4. Send the code to the Server and get a response"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "71d65e4a",
   "metadata": {},
   "outputs": [],
   "source": [
    "codes = [{\"proof\": lean_code, \"custom_id\": str(uuid.uuid4())}]\n",
    "\n",
    "response = client.verify(codes, timeout=10, infotree_type=\"original\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "babf6ec2",
   "metadata": {},
   "source": [
    "### 2.5. Extract the data (tactics and tactic states) from the infotree"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8cb4fad8",
   "metadata": {},
   "source": [
    "First, get the infotree from the response."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "2a365db7",
   "metadata": {},
   "outputs": [],
   "source": [
    "infotree = response['results'][0]['response']['infotree']"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4f1dfbc7",
   "metadata": {},
   "source": [
    "Let's first split the Lean code into header (imports) and body (statement and proof). The positions in the infotree returned by the server correspond to the positions in the body (not the full code), which is why we first have to retrieve the body in order to extract data from the infotree.\n",
    "\n",
    "Then, we can extract the tactics and tactic states from the infotree."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "c84f879f",
   "metadata": {},
   "outputs": [],
   "source": [
    "header, body = split_proof_header(lean_code)\n",
    "\n",
    "intervals = extract_data(infotree, body)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "34b566c8",
   "metadata": {},
   "source": [
    "We can now print the results."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "81fae3d8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "by\n",
      "  have h₁ : 0 < a + b + c :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ 0 < a + b + c\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ 0 < a + b + c\n",
      "---\n",
      "tactic:\n",
      " by linarith\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  have h₂ : 0 < a * b :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ 0 < a * b\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ 0 < a * b\n",
      "---\n",
      "tactic:\n",
      " by positivity\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  have h₃ : 0 < a * c :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ 0 < a * c\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ 0 < a * c\n",
      "---\n",
      "tactic:\n",
      " by positivity\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  have h₄ : 0 < b * c :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ 0 < b * c\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ 0 < b * c\n",
      "---\n",
      "tactic:\n",
      " by positivity\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  field_simp [h₁.ne', h₂.ne', h₃.ne', h₄.ne']\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) / 9 ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) / ((b + c) * (a + c) * (a + b))\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) / 9 ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) / ((b + c) * (a + c) * (a + b))\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  rw [div_le_div_iff]\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) * ((b + c) * (a + c) * (a + b)) ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) * 9\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) * ((b + c) * (a + c) * (a + b)) ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) * 9\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  nlinarith [sq_nonneg (a - b), sq_nonneg (a - 4 * c), sq_nonneg (b - 4 * c), sq_nonneg (a + b - 4 * c),\n",
      "    sq_nonneg (a + b - 2 * c), sq_nonneg (a + b + 2 * c)]\n",
      "---\n",
      "goalsAfter:\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  all_goals nlinarith\n",
      "---\n",
      "goalsAfter:\n",
      "\n",
      "--------------------\n"
     ]
    }
   ],
   "source": [
    "for interval in intervals:\n",
    "    print(\"goalsBefore:\")\n",
    "    print(\"\\n\".join(goal for goal in interval[\"goalsBefore\"]))\n",
    "    print(\"-\" * 3)\n",
    "    print(\"tactic:\")\n",
    "    print(interval[\"tactic\"])\n",
    "    print(\"-\" * 3)\n",
    "    print(\"goalsAfter:\")\n",
    "    print(\"\\n\".join(goal for goal in interval[\"goalsAfter\"]))\n",
    "    print(\"-\" * 20)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5cb5b402",
   "metadata": {},
   "source": [
    "## 3. Interaction\n",
    "\n",
    "This third section demonstrates how to use the Kimina Lean Server to interact with Lean 4 and obtain proof data (tactics and tactic states) from the server. This can be useful when implementing a tree search algorithm with a proof completion model.\n",
    "\n",
    "### 3.1. Import libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7c65f544",
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "import nest_asyncio\n",
    "import uuid\n",
    "\n",
    "sys.path.insert(0, \"YOUR_PATH_TO_kimina-lean-server\")\n",
    "\n",
    "from client.client import Lean4Client\n",
    "from client.infotree import extract_data\n",
    "from utils.proof_utils import split_proof_header, parse_client_response\n",
    "\n",
    "nest_asyncio.apply()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "631c1350",
   "metadata": {},
   "source": [
    "### 3.2. Initialize a client"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "deed5d9f",
   "metadata": {},
   "outputs": [],
   "source": [
    "client = Lean4Client(base_url=\"http://127.0.0.1:12332\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "136651e0",
   "metadata": {},
   "source": [
    "### 3.3. Load the statement"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "105461bd",
   "metadata": {},
   "source": [
    "Let's load the Lean code from a file. The code should contain the imports, a statement and a `sorry`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "d896fa80",
   "metadata": {},
   "outputs": [],
   "source": [
    "lean_file = \"demo_search_1.lean\"\n",
    "with open(lean_file, \"r\") as f:\n",
    "    lean_code = f.read()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "af8fd319",
   "metadata": {},
   "source": [
    "For this demo, we will use the following statement from Lean Workbook."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "c20815c2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "import Mathlib\n",
      "import Aesop\n",
      "\n",
      "set_option maxHeartbeats 0\n",
      "\n",
      "open BigOperators Real Nat Topology Rat\n",
      "\n",
      "/- Let $a,b,c>0$ . Prove that: $\\frac{a^{2}}{b+c}+\\frac{b^{2}}{a+c}+\\frac{16c^{2}}{a+b}\\geq \\frac{1}{9}(64c-a-b)$ -/\n",
      "theorem lean_workbook_10058 (a b c : ℝ) (ha : 0 < a) (hb : 0 < b) (hc : 0 < c) : (a^2 / (b + c) + b^2 / (a + c) + 16 * c^2 / (a + b)) ≥ (1 / 9) * (64 * c - a - b)  := by\n",
      "  sorry\n"
     ]
    }
   ],
   "source": [
    "print(lean_code)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6a445e16",
   "metadata": {},
   "source": [
    "### 3.4. Send the code to the server and get a response"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "809e7870",
   "metadata": {},
   "outputs": [],
   "source": [
    "codes = [{\"proof\": lean_code, \"custom_id\": str(uuid.uuid4())}]\n",
    "\n",
    "response = client.verify(codes, timeout=10, infotree_type=\"original\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "21cbb837",
   "metadata": {},
   "source": [
    "### 3.5. Extract the initial tactic state from the infotree"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5f57a667",
   "metadata": {},
   "source": [
    "This is the exact same process as in the data extraction example."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "b5e31603",
   "metadata": {},
   "outputs": [],
   "source": [
    "infotree = response['results'][0]['response']['infotree']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "3a675c2f",
   "metadata": {},
   "outputs": [],
   "source": [
    "header, body = split_proof_header(lean_code)\n",
    "\n",
    "intervals = extract_data(infotree, body)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "15dc167c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "by\n",
      "  sorry\n",
      "---\n",
      "goalsAfter:\n",
      "\n",
      "--------------------\n"
     ]
    }
   ],
   "source": [
    "for interval in intervals:\n",
    "    print(\"goalsBefore:\")\n",
    "    print(\"\\n\".join(goal for goal in interval[\"goalsBefore\"]))\n",
    "    print(\"-\" * 3)\n",
    "    print(\"tactic:\")\n",
    "    print(interval[\"tactic\"])\n",
    "    print(\"-\" * 3)\n",
    "    print(\"goalsAfter:\")\n",
    "    print(\"\\n\".join(goal for goal in interval[\"goalsAfter\"]))\n",
    "    print(\"-\" * 20)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "50fe6b46",
   "metadata": {},
   "source": [
    "We now have access to the initial tactic state, which can be used with the statement as input for a proof completion model."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "05d1ce63",
   "metadata": {},
   "source": [
    "### 3.6. Send a first proof attempt to the server\n",
    "\n",
    "We'll send a first proof attempt to the server. This proof attempt can be generated by a language model, prompted with the theorem statement and the initial tactic state. \n",
    "\n",
    "For this demo, let's load a proof attempt from a file."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "e33730eb",
   "metadata": {},
   "outputs": [],
   "source": [
    "lean_file = \"demo_search_2.lean\"\n",
    "with open(lean_file, \"r\") as f:\n",
    "    attempt_1 = f.read()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9fd80b4d",
   "metadata": {},
   "source": [
    "Here is our first proof attempt."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "90c58e60",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "import Mathlib\n",
      "import Aesop\n",
      "\n",
      "set_option maxHeartbeats 0\n",
      "\n",
      "open BigOperators Real Nat Topology Rat\n",
      "\n",
      "/- Let $a,b,c>0$ . Prove that: $\\frac{a^{2}}{b+c}+\\frac{b^{2}}{a+c}+\\frac{16c^{2}}{a+b}\\geq \\frac{1}{9}(64c-a-b)$ -/\n",
      "theorem lean_workbook_10058 (a b c : ℝ) (ha : 0 < a) (hb : 0 < b) (hc : 0 < c) : (a^2 / (b + c) + b^2 / (a + c) + 16 * c^2 / (a + b)) ≥ (1 / 9) * (64 * c - a - b)  := by\n",
      "  have h₁ : 0 < a + b + c := by linarith\n",
      "  have h₂ : 0 < a * b := by positivity\n",
      "  have h₃ : 0 < a * c := by positivity\n",
      "  have h₄ : 0 < b * c := by positivity\n",
      "  field_simp [h₁.ne', h₂.ne', h₃.ne', h₄.ne']\n",
      "  rw [div_le_div_iff]\n",
      "  nlinarith [ha, hb, hc]\n"
     ]
    }
   ],
   "source": [
    "print(attempt_1)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "19c85866",
   "metadata": {},
   "source": [
    "Let's send it to the server."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "b1bd2cc1",
   "metadata": {},
   "outputs": [],
   "source": [
    "codes = [{\"proof\": attempt_1, \"custom_id\": str(uuid.uuid4())}]\n",
    "\n",
    "response = client.verify(codes, timeout=10, infotree_type=\"original\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6e0a2639",
   "metadata": {},
   "source": [
    "And let's now have a look at the response. First, we can check the verification results."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "e899924c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'has_error': True, 'is_valid_no_sorry': False, 'is_valid_with_sorry': False}"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "parse_client_response(response['results'][0])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ddc0639d",
   "metadata": {},
   "source": [
    "As we can see, the proof is not valid. Let's check the messages."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "aa63649f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[{'severity': 'warning', 'pos': {'line': 13, 'column': 6}, 'endPos': {'line': 13, 'column': 20}, 'data': '`div_le_div_iff` has been deprecated: use `div_le_div_iff₀` instead'}, {'severity': 'error', 'pos': {'line': 14, 'column': 2}, 'endPos': {'line': 14, 'column': 24}, 'data': 'linarith failed to find a contradiction\\ncase a\\na b c : ℝ\\nha : 0 < a\\nhb : 0 < b\\nhc : 0 < c\\nh₁ : 0 < a + b + c\\nh₂ : 0 < a * b\\nh₃ : 0 < a * c\\nh₄ : 0 < b * c\\na✝ :\\n  (64 * c - a - b) * ((b + c) * (a + c) * (a + b)) >\\n    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) * 9\\n⊢ False failed'}]\n"
     ]
    }
   ],
   "source": [
    "messages = response['results'][0]['response']['messages']\n",
    "\n",
    "print(messages)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b540b655",
   "metadata": {},
   "source": [
    "We have two messages:\n",
    "- The first one is a warning message, indicating that `div_le_div_iff` has been deprecated. It is not an error message.\n",
    "- The second one is an error message: it indicates that the `linarith` tactic from line 21 failed.\n",
    "\n",
    "Let's have a look at the tactics and tactic states extracted from the infotree."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "d6c531d1",
   "metadata": {},
   "outputs": [],
   "source": [
    "infotree = response['results'][0]['response']['infotree']\n",
    "\n",
    "header, body = split_proof_header(attempt_1)\n",
    "\n",
    "intervals = extract_data(infotree, body)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "120e3448",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "by\n",
      "  have h₁ : 0 < a + b + c :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ 0 < a + b + c\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ 0 < a + b + c\n",
      "---\n",
      "tactic:\n",
      " by linarith\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  have h₂ : 0 < a * b :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ 0 < a * b\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ 0 < a * b\n",
      "---\n",
      "tactic:\n",
      " by positivity\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  have h₃ : 0 < a * c :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ 0 < a * c\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ 0 < a * c\n",
      "---\n",
      "tactic:\n",
      " by positivity\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  have h₄ : 0 < b * c :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ 0 < b * c\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ 0 < b * c\n",
      "---\n",
      "tactic:\n",
      " by positivity\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  field_simp [h₁.ne', h₂.ne', h₃.ne', h₄.ne']\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) / 9 ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) / ((b + c) * (a + c) * (a + b))\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) / 9 ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) / ((b + c) * (a + c) * (a + b))\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  rw [div_le_div_iff]\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) * ((b + c) * (a + c) * (a + b)) ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) * 9\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) * ((b + c) * (a + c) * (a + b)) ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) * 9\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  nlinarith [ha, hb, hc]\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) * ((b + c) * (a + c) * (a + b)) ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) * 9\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "--------------------\n"
     ]
    }
   ],
   "source": [
    "for interval in intervals:\n",
    "    print(\"goalsBefore:\")\n",
    "    print(\"\\n\".join(goal for goal in interval[\"goalsBefore\"]))\n",
    "    print(\"-\" * 3)\n",
    "    print(\"tactic:\")\n",
    "    print(interval[\"tactic\"])\n",
    "    print(\"-\" * 3)\n",
    "    print(\"goalsAfter:\")\n",
    "    print(\"\\n\".join(goal for goal in interval[\"goalsAfter\"]))\n",
    "    print(\"-\" * 20)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "92c93f60",
   "metadata": {},
   "source": [
    "### 3.7. Send a new proof attempt\n",
    "\n",
    "We can reconstruct the proof attempt from the extracted tactics, stopping right before the failed tactic. From the message, we know that the last tactic failed. The correct tactics can be concatenated to give the start of our proof attempt."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "b2f5a1cd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "by\n",
      "  have h₁ : 0 < a + b + c := by linarith\n",
      "  have h₂ : 0 < a * b := by positivity\n",
      "  have h₃ : 0 < a * c := by positivity\n",
      "  have h₄ : 0 < b * c := by positivity\n",
      "  field_simp [h₁.ne', h₂.ne', h₃.ne', h₄.ne']\n",
      "  rw [div_le_div_iff]\n"
     ]
    }
   ],
   "source": [
    "current_proof = \"\".join(intervals[i][\"tactic\"] for i in range(len(intervals) - 1))\n",
    "\n",
    "print(current_proof)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c1431623",
   "metadata": {},
   "source": [
    "We can also have access to the last tactic state before the failed tactic."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "ef1bd656",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) * ((b + c) * (a + c) * (a + b)) ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) * 9\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n"
     ]
    }
   ],
   "source": [
    "print(\"\\n\".join(goal for goal in intervals[-1][\"goalsBefore\"]))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "637967c5",
   "metadata": {},
   "source": [
    "We can now generate a new proof attempt. This proof attempt can be generated by a language model, prompted with the theorem statement, the current proof, and the current tactic state. \n",
    "\n",
    "For this demo, let's load a new proof attempt from a file."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "8ceb2c47",
   "metadata": {},
   "outputs": [],
   "source": [
    "lean_file = \"demo_search_3.lean\"\n",
    "with open(lean_file, \"r\") as f:\n",
    "    attempt_2 = f.read()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "3e6d21b1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "import Mathlib\n",
      "import Aesop\n",
      "\n",
      "set_option maxHeartbeats 0\n",
      "\n",
      "open BigOperators Real Nat Topology Rat\n",
      "\n",
      "/- Let $a,b,c>0$ . Prove that: $\\frac{a^{2}}{b+c}+\\frac{b^{2}}{a+c}+\\frac{16c^{2}}{a+b}\\geq \\frac{1}{9}(64c-a-b)$ -/\n",
      "theorem lean_workbook_10058 (a b c : ℝ) (ha : 0 < a) (hb : 0 < b) (hc : 0 < c) : (a^2 / (b + c) + b^2 / (a + c) + 16 * c^2 / (a + b)) ≥ (1 / 9) * (64 * c - a - b)  := by\n",
      "  have h₁ : 0 < a + b + c := by linarith\n",
      "  have h₂ : 0 < a * b := by positivity\n",
      "  have h₃ : 0 < a * c := by positivity\n",
      "  have h₄ : 0 < b * c := by positivity\n",
      "  field_simp [h₁.ne', h₂.ne', h₃.ne', h₄.ne']\n",
      "  rw [div_le_div_iff]\n",
      "  nlinarith [sq_nonneg (a - b), sq_nonneg (a - 4 * c), sq_nonneg (b - 4 * c), sq_nonneg (a + b - 4 * c),\n",
      "    sq_nonneg (a + b - 2 * c), sq_nonneg (a + b + 2 * c)]\n",
      "  all_goals nlinarith\n",
      "\n"
     ]
    }
   ],
   "source": [
    "print(attempt_2)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2c3e25e8",
   "metadata": {},
   "source": [
    "Let's do the same as before: send it to the server and have a look at the response."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "5c53ef20",
   "metadata": {},
   "outputs": [],
   "source": [
    "codes = [{\"proof\": attempt_2, \"custom_id\": str(uuid.uuid4())}]\n",
    "\n",
    "response = client.verify(codes, timeout=10, infotree_type=\"original\")\n",
    "\n",
    "infotree = response['results'][0]['response']['infotree']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "014f3aec",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'has_error': False, 'is_valid_no_sorry': True, 'is_valid_with_sorry': True}"
      ]
     },
     "execution_count": 37,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "parse_client_response(response['results'][0])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5d4ae9cc",
   "metadata": {},
   "source": [
    "This time, the proof seems correct!\n",
    "\n",
    "Let's have a look at the extracted tactics and tactic states."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "85c5a204",
   "metadata": {},
   "outputs": [],
   "source": [
    "header, body = split_proof_header(attempt_2)\n",
    "\n",
    "intervals = extract_data(infotree, body)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "6f4c1c18",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "by\n",
      "  have h₁ : 0 < a + b + c :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ 0 < a + b + c\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "⊢ 0 < a + b + c\n",
      "---\n",
      "tactic:\n",
      " by linarith\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  have h₂ : 0 < a * b :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ 0 < a * b\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "⊢ 0 < a * b\n",
      "---\n",
      "tactic:\n",
      " by positivity\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  have h₃ : 0 < a * c :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ 0 < a * c\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "⊢ 0 < a * c\n",
      "---\n",
      "tactic:\n",
      " by positivity\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  have h₄ : 0 < b * c :=\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ 0 < b * c\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "⊢ 0 < b * c\n",
      "---\n",
      "tactic:\n",
      " by positivity\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ a ^ 2 / (b + c) + b ^ 2 / (a + c) + 16 * c ^ 2 / (a + b) ≥ 1 / 9 * (64 * c - a - b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  field_simp [h₁.ne', h₂.ne', h₃.ne', h₄.ne']\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) / 9 ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) / ((b + c) * (a + c) * (a + b))\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) / 9 ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) / ((b + c) * (a + c) * (a + b))\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  rw [div_le_div_iff]\n",
      "---\n",
      "goalsAfter:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) * ((b + c) * (a + c) * (a + b)) ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) * 9\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ (64 * c - a - b) * ((b + c) * (a + c) * (a + b)) ≤\n",
      "    ((a ^ 2 * (a + c) + b ^ 2 * (b + c)) * (a + b) + 16 * c ^ 2 * ((b + c) * (a + c))) * 9\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  nlinarith [sq_nonneg (a - b), sq_nonneg (a - 4 * c), sq_nonneg (b - 4 * c), sq_nonneg (a + b - 4 * c),\n",
      "    sq_nonneg (a + b - 2 * c), sq_nonneg (a + b + 2 * c)]\n",
      "---\n",
      "goalsAfter:\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "--------------------\n",
      "goalsBefore:\n",
      "case b0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < 9\n",
      "case d0\n",
      "a b c : ℝ\n",
      "ha : 0 < a\n",
      "hb : 0 < b\n",
      "hc : 0 < c\n",
      "h₁ : 0 < a + b + c\n",
      "h₂ : 0 < a * b\n",
      "h₃ : 0 < a * c\n",
      "h₄ : 0 < b * c\n",
      "⊢ 0 < (b + c) * (a + c) * (a + b)\n",
      "---\n",
      "tactic:\n",
      "\n",
      "  all_goals nlinarith\n",
      "---\n",
      "goalsAfter:\n",
      "\n",
      "--------------------\n"
     ]
    }
   ],
   "source": [
    "for interval in intervals:\n",
    "    print(\"goalsBefore:\")\n",
    "    print(\"\\n\".join(goal for goal in interval[\"goalsBefore\"]))\n",
    "    print(\"-\" * 3)\n",
    "    print(\"tactic:\")\n",
    "    print(interval[\"tactic\"])\n",
    "    print(\"-\" * 3)\n",
    "    print(\"goalsAfter:\")\n",
    "    print(\"\\n\".join(goal for goal in interval[\"goalsAfter\"]))\n",
    "    print(\"-\" * 20)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
