{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "027e03c0-2507-4356-b0e7-eaf3a9e2e8c1",
   "metadata": {},
   "outputs": [],
   "source": [
    "import argparse\n",
    "from dataclasses import asdict\n",
    "import json\n",
    "import os\n",
    "\n",
    "import torch\n",
    "from DatasetInit import CustomDataset\n",
    "from DatasetJudge import DatasetJudge\n",
    "from get_hidden_state import get_hidden_state, InputQA\n",
    "from get_known import get_isKnown\n",
    "from mlp_trainer import TrainResult, train\n",
    "from utils import jsonlload, load_model, seed_everything, get_proj, split_data\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "plt.rcParams['axes.unicode_minus'] = False  # -"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "af929af2-0432-4e38-8639-1135acfcf768",
   "metadata": {},
   "outputs": [],
   "source": [
    "import functools\n",
    "import time\n",
    "from typing import Callable\n",
    "import psutil\n",
    "\n",
    "def measure_resources_multigpu(func: Callable):\n",
    "    @functools.wraps(func)\n",
    "    def wrapper(*args, **kwargs):\n",
    "        # Initial resource recording (CPU and GPU)\n",
    "        process = psutil.Process()\n",
    "        initial_cpu_memory = process.memory_info().rss\n",
    "        \n",
    "        gpu_initial_stats = {}\n",
    "        if torch.cuda.is_available():\n",
    "            torch.cuda.synchronize()\n",
    "            # Record initial state for each GPU\n",
    "            for i in range(torch.cuda.device_count()):\n",
    "                gpu_initial_stats[i] = {\n",
    "                    'allocated': torch.cuda.memory_allocated(i),\n",
    "                    'cached': torch.cuda.memory_reserved(i)  # Note: In older PyTorch versions, 'reserved' might be called 'cached'\n",
    "                }\n",
    "            torch.cuda.reset_peak_memory_stats()  # Reset peak statistics\n",
    "\n",
    "        # Time measurement\n",
    "        start_time = time.time()\n",
    "        result = func(*args, **kwargs)  # Execute function\n",
    "        end_time = time.time()\n",
    "\n",
    "        # Calculate resource consumption\n",
    "        cpu_memory_used = (process.memory_info().rss - initial_cpu_memory) / 1024**3  # Convert to GB\n",
    "        \n",
    "        gpu_memory_used = {}\n",
    "        gpu_peak_memory = {}\n",
    "        gpu_peak_add = {}\n",
    "        if torch.cuda.is_available():\n",
    "            torch.cuda.synchronize()\n",
    "            for i in range(torch.cuda.device_count()):\n",
    "                current_allocated = torch.cuda.memory_allocated(i)\n",
    "                # Calculate memory increment for current GPU\n",
    "                gpu_memory_used[i] = (current_allocated - gpu_initial_stats[i]['allocated']) / 1024**3\n",
    "                # Get peak memory for current GPU\n",
    "                gpu_peak_memory[i] = torch.cuda.max_memory_allocated(i) / 1024**3\n",
    "                gpu_peak_add[i] = (torch.cuda.max_memory_allocated(i) - gpu_initial_stats[i]['allocated']) / 1024**3\n",
    "\n",
    "        # Print results\n",
    "        print(f\"\\n{'='*50}\")\n",
    "        print(f\"Function: {func.__name__}\")\n",
    "        print(f\"Execution time: {end_time - start_time:.2f} seconds\")\n",
    "        print(f\"CPU memory increment: {cpu_memory_used:.2f} GB\")\n",
    "        if torch.cuda.is_available():\n",
    "            for i in range(torch.cuda.device_count()):\n",
    "                print(f\"GPU {i} - Memory increment: {gpu_memory_used[i]:.2f} GB, Peak memory: {gpu_peak_memory[i]:.2f} GB, Peak increment: {gpu_peak_add[i]:.2f} GB\")\n",
    "        print(f\"{'='*50}\\n\")\n",
    "        \n",
    "        return result\n",
    "    return wrapper"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "4aababf4-7d47-44bc-91ef-08152147bb52",
   "metadata": {},
   "outputs": [],
   "source": [
    "MODEL_PATH = {\n",
    "    \"Qwen2.5-7B-Instruct\": \"Qwen/Qwen2.5-7B-Instruct\",\n",
    "    \"BLEURT-20\": \"lucadiliello/BLEURT-20\",\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "2da330da-dd04-46dc-80a8-4cce97b5ff37",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model_quantity = \"Llama-3.1-8B\"\n",
    "model_quantity = \"Qwen2.5-7B-Instruct\"\n",
    "# model_quantity = \"Qwen2.5-72B-Instruct\"\n",
    "# model_quantity = \"Qwen3-235B-A22B-Instruct\"\n",
    "# model_quantity = \"Qwen2.5-0.5B-Instruct\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "a6cfd093-a33c-48be-929d-a343097a56e2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Loading model: Qwen2.5-7B-Instruct...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "`torch_dtype` is deprecated! Use `dtype` instead!\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d73a614b52954011951c6d082ff71696",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "Loading checkpoint shards:   0%|          | 0/4 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "==================================================\n",
      "Function: init\n",
      "Execution time: 9.09 seconds\n",
      "CPU memory increment: 0.30 GB\n",
      "GPU 0 - Memory increment: 6.22 GB, Peak memory: 6.22 GB, Peak increment: 6.22 GB\n",
      "GPU 1 - Memory increment: 7.96 GB, Peak memory: 7.96 GB, Peak increment: 7.96 GB\n",
      "==================================================\n",
      "\n"
     ]
    }
   ],
   "source": [
    "@measure_resources_multigpu\n",
    "def init():\n",
    "    print(f\"Loading model: {model_quantity}...\")\n",
    "    return load_model(\n",
    "        MODEL_PATH[model_quantity],\n",
    "        torch_dtype=\"auto\",\n",
    "        device_map=\"auto\",\n",
    "        attn_implementation=\"eager\",\n",
    "    )\n",
    "    \n",
    "model, tokenizer = init()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "2d82625f-50f6-4fcf-92a8-b233a11237ed",
   "metadata": {},
   "outputs": [],
   "source": [
    "layer_num = model.config.num_hidden_layers\n",
    "vocab_size = model.config.vocab_size\n",
    "# data_device = 'cuda:0'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "9c3b499d-e3e8-4afd-93c8-187d00234c02",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "==================================================\n",
      "Function: compute_svd\n",
      "Execution time: 1.60 seconds\n",
      "CPU memory increment: 0.17 GB\n",
      "GPU 0 - Memory increment: 0.00 GB, Peak memory: 6.22 GB, Peak increment: 0.00 GB\n",
      "GPU 1 - Memory increment: 0.02 GB, Peak memory: 16.34 GB, Peak increment: 8.37 GB\n",
      "==================================================\n",
      "\n"
     ]
    }
   ],
   "source": [
    "@measure_resources_multigpu\n",
    "def compute_svd(w):\n",
    "    with torch.no_grad():\n",
    "        original_dtype = w.dtype\n",
    "        if original_dtype == torch.float32:\n",
    "            svd_res = torch.svd(w)\n",
    "            V = svd_res.V\n",
    "            S = svd_res.S\n",
    "        else:\n",
    "            svd_res = torch.svd(w.float())\n",
    "            V = svd_res.V.to(original_dtype)\n",
    "            S = svd_res.S.to(original_dtype)\n",
    "    return S, V\n",
    "\n",
    "S, V = compute_svd(model.lm_head.weight.data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "e4dd56a9-4adc-493a-b800-196712a40c66",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([152064, 3584])"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.lm_head.weight.data.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "a2e024d5-fdb6-4517-a221-43bb3cd2bc8a",
   "metadata": {},
   "outputs": [],
   "source": [
    "proj_dim = 256\n",
    "proj_base = V[:, -proj_dim:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "cb428849-4e77-43ed-96e0-e58f36d239f5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'Answer the following question step by step to the best of your ability.\\nQ: Charlie wants to sell beeswax candles. For every pound of beeswax, he can make 10 tapered candles. One pound of beeswax and the wicks cost $10.00 in supplies. If he sells each candle for $2.00 each, what is his net profit if he makes and sells 20 candles?\\nA: '"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "QA = \"\"\"Q: Charlie wants to sell beeswax candles. For every pound of beeswax, he can make 10 tapered candles. One pound of beeswax and the wicks cost $10.00 in supplies. If he sells each candle for $2.00 each, what is his net profit if he makes and sells 20 candles?\n",
    "A: \"\"\"\n",
    "# QA = \"\"\"Q: Mishka bought 3 pairs of shorts, 3 pairs of pants, and 3 pairs of shoes. One pair of shorts costs $16.50. One pair of pants costs $22.50 and one pair of shoes costs $42. How many dollars did Mishka spend on all the clothing items?\n",
    "# A: \"\"\"\n",
    "CoT_QA = f\"\"\"Answer the following question step by step to the best of your ability.\\n{QA}\"\"\"\n",
    "# CoT_QA = f\"\"\"{QA}Let's think step by step.\\n\"\"\"\n",
    "\n",
    "# CoT_QA = f\"\"\"Q: A robe takes 2 bolts of blue fiber and half that much white fiber. How many bolts in total does it take?\n",
    "# A: A robe needs 2 bolts of blue fiber.\n",
    "# The amount of white fiber needed is half of the blue fiber.\n",
    "# Half of 2 bolts is 1 bolt of white fiber.\n",
    "# The total bolts needed is the sum of blue and white fiber.\n",
    "# 2 bolts plus 1 bolt equals 3 bolts.\n",
    "# Therefore, the final answer is 3.\n",
    "\n",
    "# Q: James decides to run 3 sprints 3 times a week. He runs 60 meters each sprint. How many total meters does he run a week?\n",
    "# A: James runs 3 sprints each time he exercises.\n",
    "# Each sprint is 60 meters.\n",
    "# In one exercise session, he runs 3 × 60 = 180 meters.\n",
    "# He exercises 3 times per week.\n",
    "# The total distance he runs in a week is 180 × 3 = 540 meters.\n",
    "# Therefore, the final answer is 540.\n",
    "\n",
    "# {QA}\"\"\"\n",
    "\n",
    "# CoT_QA = f\"\"\"Q: A robe takes 2 bolts of blue fiber and half that much white fiber. How many bolts in total does it take?\n",
    "# A: A robe needs 2 bolts of blue fiber.\n",
    "# The amount of white fiber needed is half of the blue fiber.\n",
    "# Half of 2 bolts is 1 bolt of white fiber.\n",
    "# The total bolts needed is the sum of blue and white fiber.\n",
    "# 2 bolts plus 1 bolt equals 3 bolts.\n",
    "# Therefore, the final answer is 3.\n",
    "\n",
    "# {QA}\"\"\"\n",
    "\n",
    "CoT_QA"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "6a3ec137-c903-4679-84cb-64bd9d10d5e2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'Q: Charlie wants to sell beeswax candles. For every pound of beeswax, he can make 10 tapered candles. One pound of beeswax and the wicks cost $10.00 in supplies. If he sells each candle for $2.00 each, what is his net profit if he makes and sells 20 candles?\\nA: '"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "t_qa = tokenizer.encode(QA)\n",
    "t_cot = tokenizer.encode(CoT_QA)\n",
    "tokenizer.decode(t_cot[-len(t_qa):])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "ce95f5f9-69ee-4672-9142-37dd2f533b90",
   "metadata": {},
   "outputs": [],
   "source": [
    "from utils import ModelHook\n",
    "key_list = [f\"model.layers.{layer}\" for layer in range(layer_num - 1)]\n",
    "# key_list += [f\"model.layers.{layer}.self_attn\" for layer in range(layer_num)]\n",
    "# key_list += [f\"model.layers.{layer}.mlp\" for layer in range(layer_num)]\n",
    "\n",
    "hooked_model = ModelHook(model, tokenizer)\n",
    "hooked_model.remove_all_hook()\n",
    "hooked_handle = hooked_model.hook_model_input_output(hook_name_list=key_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "3848fc16-09a5-4070-885f-f556d29a9438",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "hooked_model.clear_io()\n",
    "CoT_result = hooked_model.generate_with_hook(\n",
    "    CoT_QA,\n",
    "    max_new_tokens=512,\n",
    "    top_k=1\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "a3973fee-2b6d-4a32-be95-812998977c82",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_reasoning_component(emb, proj_base):\n",
    "    p = emb  @ proj_base\n",
    "    h_reasoning = p @ proj_base.T\n",
    "    return h_reasoning"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "216a9b6d-8f71-445d-b26a-9ccfd488c7d2",
   "metadata": {},
   "outputs": [],
   "source": [
    "CoT_result['h_reasoning'] = {}\n",
    "\n",
    "for key in key_list:\n",
    "    CoT_result['h_reasoning'][key] = []\n",
    "    for emb in CoT_result['hook_output_dict'][key]:\n",
    "        CoT_result['h_reasoning'][key].append(\n",
    "            get_reasoning_component(emb, proj_base.to(emb.device))\n",
    "        )\n",
    "    \n",
    "# CoT_result['h_reasoning']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "d2902bdb-a380-4c8f-8362-269f1533d251",
   "metadata": {},
   "outputs": [],
   "source": [
    "hooked_model.clear_io()\n",
    "\n",
    "QA_result = hooked_model.generate_with_hook(\n",
    "    QA,\n",
    "    max_new_tokens=512,\n",
    "    top_k=1\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "6a299318-53ee-4277-b5aa-3e72f7f16e13",
   "metadata": {},
   "outputs": [],
   "source": [
    "QA_result['h_reasoning'] = {}\n",
    "\n",
    "for key in key_list:\n",
    "    QA_result['h_reasoning'][key] = []\n",
    "    for emb in QA_result['hook_output_dict'][key]:\n",
    "        QA_result['h_reasoning'][key].append(\n",
    "            get_reasoning_component(emb, proj_base.to(emb.device))\n",
    "        )\n",
    "    \n",
    "# QA_result['h_reasoning']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "034b996a-9eaa-4882-8a3f-3653e7f93703",
   "metadata": {},
   "outputs": [],
   "source": [
    "check = {key: 0 for key in key_list}\n",
    "max_change_num = 1  # only once for input question\n",
    "def _hook_change_reasoning_component(module, input, output):\n",
    "    name = hooked_model.module2name[module]\n",
    "    if check[name] >= max_change_num:\n",
    "        return output\n",
    "    h_reasoning = CoT_result['h_reasoning'][name][check[name]]\n",
    "    check[name] += 1\n",
    "    \n",
    "    if isinstance(output, tuple):\n",
    "        all_data = output[0]\n",
    "    else:\n",
    "        all_data = output\n",
    "    bsz, seq_len, hidden_size = all_data.size()\n",
    "\n",
    "    for i in range(bsz):\n",
    "        for j in range(seq_len):\n",
    "            emb = all_data[i, j, :]    \n",
    "            emb = emb - get_reasoning_component(emb, proj_base.to(emb.device)) + h_reasoning[i, -seq_len + j, :].to(emb.device)\n",
    "\n",
    "            if isinstance(output, tuple):\n",
    "                output[0][i, j, :] = emb\n",
    "            else:\n",
    "                output[i, j, :] = emb\n",
    "    return output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "dd8361df-085e-4653-866f-7d36cb51e8a6",
   "metadata": {},
   "outputs": [],
   "source": [
    "hooked_model.remove_all_hook()\n",
    "hooked_handle = hooked_model.hook_model_input_output(hook_name_list=key_list)\n",
    "# add change hook\n",
    "check = {key: 0 for key in key_list}\n",
    "max_change_num = 1  # only once for input question\n",
    "cur_handle = []\n",
    "for name, module in hooked_model.model.named_modules():\n",
    "    if name in key_list:\n",
    "        handle = module.register_forward_hook(_hook_change_reasoning_component)\n",
    "        cur_handle.append(handle)\n",
    "hooked_model.hook_handle_list.extend(cur_handle)\n",
    "if cur_handle:\n",
    "    hooked_model.is_hook = True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "035ab8b1-75ae-485f-9469-1e1a4f5ecf68",
   "metadata": {},
   "outputs": [],
   "source": [
    "hooked_model.clear_io()\n",
    "\n",
    "change_QA_result = hooked_model.generate_with_hook(\n",
    "    QA,\n",
    "    max_new_tokens=512,\n",
    "    top_k=1\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "e7bf1c4d-d58b-40ed-bef8-d2dca0f02315",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "change_QA_result['h_reasoning'] = {}\n",
    "\n",
    "for key in key_list:\n",
    "    change_QA_result['h_reasoning'][key] = []\n",
    "    for emb in change_QA_result['hook_output_dict'][key]:\n",
    "        change_QA_result['h_reasoning'][key].append(\n",
    "            get_reasoning_component(emb, proj_base.to(emb.device))\n",
    "        )\n",
    "    \n",
    "# change_QA_result['h_reasoning']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "0a5a04b0-d826-44ed-9437-fd50efbdec97",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1. First, let's calculate the total cost of supplies for making 20 candles:\n",
      "   - We know that one pound of beeswax and a wick cost $11.00\n",
      "   - To find out how many pounds of beeswax are needed for 20 candles, we divide the number of candles by the number of candles per pound: 20 / 10 = 2 pounds\n",
      "   - Therefore, the total cost of supplies is 2 * $11.00 = $22.00\n",
      "\n",
      "2. Next, let's calculate the total revenue from selling 20 candles:\n",
      "   - We know that each candle is sold for $2.00\n",
      "   - Therefore, the total revenue is 20 * $2.00 = $44.00\n",
      "\n",
      "3. Finally, let's calculate the net profit:\n",
      "   - Net profit is total revenue minus total cost of supplies\n",
      "   - Net profit = $44.00 - $22.00 = $22.00\n",
      "\n",
      "Therefore, Charlie's net profit if he makes and sells 20 candles is $22.00. Answer: $22.00. Step-by-step solution: 1. Calculate the total cost of supplies for 20 candles. 2. Calculate the total revenue from selling 20 candles. 3. Calculate the net profit by subtracting the total cost of supplies from the total revenue. Final answer: $22.00.\n"
     ]
    }
   ],
   "source": [
    "print(CoT_result['response'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "f1a2cd3f-888a-4183-a577-2b8db23818bd",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "3.6\n",
      "To calculate Charlie's net profit from selling 20 candles, we need to follow these steps:\n",
      "\n",
      "1. **Determine the number of pounds of beeswax needed:**  \n",
      "   Since Charlie can make 10 tapered candles per pound of beeswax, for 20 candles, he needs:\n",
      "   \\[\n",
      "   \\frac{20 \\text{ candles}}{10 \\text{ candles/pound}} = 2 \\text{ pounds of beeswax}\n",
      "   \\]\n",
      "\n",
      "2. **Calculate the total cost of supplies:**  \n",
      "   The cost for one pound of beesw and a wick is $10.50. Therefore, for 2 pounds:\n",
      "   \\[\n",
      "   2 \\text{ pounds} \\times \\$10.50/\\text{pound} = \\$21.00\n",
      "   \\]\n",
      "\n",
      "3. **Calculate the total revenue from selling the candles:**  \n",
      "   Each candle is sold for $2.00, so for 20 candles:\n",
      "   \\[\n",
      "   20 \\text{ candles} \\times \\$2.00/\\text{candle} = \\$40.00\n",
      "   \\]\n",
      "\n",
      "4. **Calculate the net profit:**  \n",
      "   Net profit is calculated by subtracting the total cost from the total revenue:\n",
      "   \\[\n",
      "   \\$40.00 - \\$21.50 = \\$18.50\n",
      "   \\]\n",
      "\n",
      "Thus, Charlie's net profit if he makes and sells 20 candles is:\n",
      "\\[\n",
      "\\boxed{18.50}\n",
      "\\] \n",
      "\n",
      "It seems there was an error in the problem statement or the provided answer. Based on the calculations, the correct net profit should be $18.50, not $3.60. Please verify the problem details again. If the correct answer is indeed $3.60, then the numbers or conditions might have been different. Let me know if you need further clarification.\n"
     ]
    }
   ],
   "source": [
    "print(QA_result['response'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "6e69c471-6a17-4dfb-a877-ce2676e0949e",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1. Calculate the total cost of supplies for making candles:\n",
      "   \\[\n",
      "   \\text{Total cost} = 20 \\text{ candles} \\times \\$1.00 \\text{ per candle} = \\$20.00\n",
      "   \\]\n",
      "2. Calculate the total revenue from selling the candles:\n",
      "   \\[\n",
      "   \\text{Total revenue} = 20 \\text{ candles} \\times \\$2.00 \\text{ per candle} = \\$40.00\n",
      "   \\]\n",
      "3. Calculate the net profit by subtracting the total cost from the total revenue:\n",
      "   \\[\n",
      "   \\text{Net profit} = \\$40.00 - \\$20.00 = \\$20.00\n",
      "   \\]\n",
      "\n",
      "Therefore, the net profit for Charlie if he makes and sells 20 candles is:\n",
      "\\[\n",
      "\\boxed{\\$20.00}\n",
      "\\]\n"
     ]
    }
   ],
   "source": [
    "print(change_QA_result['response'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9aa42685-1dcd-4a3d-a37d-3e827a73f215",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "aa11a3ca-798f-45fa-8386-285ed0bd4b91",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "HARP",
   "language": "python",
   "name": "harp"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.19"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
