{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Data Prep Pipeline for RADT"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## reach obstacle"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Convert dataset generated for baselines -> RADT format"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "from raDT.constants import *\n",
    "# Open the pickle file in read binary mode\n",
    "with open(HOME_PATH_BASELINES + 'offline_data/random/FetchReachObstacle/buffer_massless.pkl', 'rb') as f:\n",
    "    data = pickle.load(f)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(40000, 50, 19)\n",
      "(40000, 50, 3)\n",
      "(40000, 49, 3)\n",
      "(40000, 49, 4)\n",
      "(40000, 49, 1)\n",
      "(40000, 49, 1)\n"
     ]
    }
   ],
   "source": [
    "print(data['o'].shape)\n",
    "print(data['ag'].shape)\n",
    "print(data['g'].shape)\n",
    "print(data['u'].shape)\n",
    "print(data['r'].shape)\n",
    "print(data['c'].shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_radt = []\n",
    "for i in range(data['o'].shape[0]):\n",
    "    traj = {}\n",
    "    traj[\"observations\"] = data['o'][i, :-1, :]\n",
    "    traj['next_observations'] = data['o'][i, 1:, :]\n",
    "    traj['actions'] = data['u'][i, :, :]\n",
    "    traj['timesteps'] = traj[\"observations\"].shape[0]\n",
    "    traj['rewards'] = data['r'][i, :, :].flatten()\n",
    "    traj['costs'] = data['c'][i, :, :].flatten()\n",
    "    traj['terminals'] = np.array([False] * (traj['timesteps'] - 1) + [True])\n",
    "    traj['goal_pos'] = list(data['g'][i, 0])\n",
    "    data_radt = data_radt + [traj]\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Export dataset to radt dataset directory\n",
    "with open('dataset/reach_obstacle_massless.pkl', 'wb') as handle:\n",
    "    # Load the data from the file\n",
    "    pickle.dump(data_radt, handle)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### RADT data prep (hindsight avoid region relabeling)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "import mpld3\n",
    "# calculate avoid box inference\n",
    "\n",
    "from scipy.spatial import ConvexHull\n",
    "\n",
    "from concave_hull import convex_hull, convex_hull_indexes, concave_hull, concave_hull_indexes\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "  \n",
    "# importing required libraries \n",
    "from mpl_toolkits.mplot3d import Axes3D \n",
    "import matplotlib.pyplot as plt \n",
    "import copy\n",
    "import itertools\n",
    "\n",
    "import numpy as np\n",
    "from scipy import arccos\n",
    "from scipy.linalg import norm\n",
    "\n",
    "# Open the pickle file of dataset in radt directory in read binary mode\n",
    "with open('dataset/reach_obstacle_massless.pkl', 'rb') as f:\n",
    "    # Load the data from the file\n",
    "    data_radt = pickle.load(f)\n",
    "\n",
    "data_radt_copy = copy.deepcopy(data_radt)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Code for contour-based avoid centroid sampling strategy\n",
    "\n",
    "def add_pre_avoid_states(traj):\n",
    "    sample_traj = traj[\"observations\"][:,:3]\n",
    "    concave_hull_idx = concave_hull_indexes(sample_traj, concavity = 1)\n",
    "    convex_hull_idx = convex_hull_indexes(sample_traj)\n",
    "    bool_concave = ~np.isin(concave_hull_idx, convex_hull_idx)\n",
    "    prev_group_id = 0\n",
    "    prev_item_bool = False\n",
    "    groups = []\n",
    "    for b in bool_concave:\n",
    "        if not b: \n",
    "            groups.append(0)\n",
    "            prev_item_bool = False\n",
    "        else:\n",
    "            if not prev_item_bool:\n",
    "                prev_group_id += 1\n",
    "                groups.append(prev_group_id)\n",
    "            else:\n",
    "                groups.append(prev_group_id)\n",
    "            prev_item_bool = True\n",
    "\n",
    "    groups = np.array(groups)\n",
    "\n",
    "    if bool_concave[0] and bool_concave[-1]:\n",
    "        groups[groups == prev_group_id] = 1\n",
    "\n",
    "    n = len(groups)\n",
    "    groups_dict={}\n",
    "    for group in range(1, max(groups) + 1):\n",
    "        groups_dict[group] = []\n",
    "        for i in range(n):\n",
    "            a = False\n",
    "            for p in [(i-1) % n, i % n, (i+1) % n]:\n",
    "                if groups[p] == group:\n",
    "                    a = True\n",
    "            if a and (concave_hull_idx[i] not in groups_dict[group]):\n",
    "                groups_dict[group].append(concave_hull_idx[i])\n",
    "\n",
    "    # new filtering\n",
    "    new_groups_dict = {}\n",
    "    for k,v in groups_dict.items():\n",
    "        if len(groups_dict[k]) >= 10:\n",
    "            new_groups_dict[k] = v\n",
    "    ###\n",
    "\n",
    "    groups_dict = new_groups_dict\n",
    "    groups_dict\n",
    "\n",
    "    means = []\n",
    "    for idxs in groups_dict.values():\n",
    "        concave_points = sample_traj[idxs, :]\n",
    "        means.append(np.mean(concave_points, axis = 0))\n",
    "\n",
    "    means = np.array(means)\n",
    "\n",
    "    dual_means = []\n",
    "    for idxs in groups_dict.values():\n",
    "        concave_points = sample_traj[[idxs[0],idxs[-1]], :]\n",
    "        dual_means.append(np.mean(concave_points, axis = 0))\n",
    "\n",
    "    dual_means = np.array(dual_means)\n",
    "\n",
    "    new_dual_means = []\n",
    "    for mean in dual_means:\n",
    "        v1 = sample_traj[-1] - sample_traj[0]\n",
    "        v2 = mean - sample_traj[0]\n",
    "        v3 = -v1\n",
    "        v4 = mean - sample_traj[-1]\n",
    "        v1_u = v1 / norm(v1)\n",
    "        v2_u = v2 / norm(v2)\n",
    "        v3_u = v3 / norm(v3)\n",
    "        v4_u = v4 / norm(v4)\n",
    "        angle1 = arccos(np.clip(np.dot(v1_u, v2_u), -1.0, 1.0))\n",
    "        angle2 = arccos(np.clip(np.dot(v3_u, v4_u), -1.0, 1.0))\n",
    "        if abs(angle1) <= np.pi/6 and abs(angle2) <= np.pi/6:\n",
    "            new_dual_means.append(mean)\n",
    "\n",
    "    random_pts = []\n",
    "    num_random = np.random.randint(3)\n",
    "    x_min, x_max = np.min(sample_traj[:, 0]), np.max(sample_traj[:, 0])\n",
    "    y_min, y_max = np.min(sample_traj[:, 1]), np.max(sample_traj[:, 1])\n",
    "    z_min, z_max = np.min(sample_traj[:, 2]), np.max(sample_traj[:, 2])\n",
    "\n",
    "    xranges = [(x_max + 0.04, x_max + 0.2), (x_min - 0.2, x_min - 0.04)]\n",
    "    yranges = [(y_max + 0.04, y_max + 0.2), (y_min - 0.2, y_min - 0.04)]\n",
    "    zranges = [(z_max + 0.04, z_max + 0.2), (z_min - 0.2, z_min - 0.04)]\n",
    "\n",
    "\n",
    "    for i in range(num_random):\n",
    "        x_range = xranges[np.random.randint(2)]\n",
    "        y_range = yranges[np.random.randint(2)]\n",
    "        z_range = zranges[np.random.randint(2)]\n",
    "        x, y, z= np.random.uniform(*x_range), np.random.uniform(*y_range), np.random.uniform(*z_range)\n",
    "        random_pts.append(np.array([x, y, z]))\n",
    "\n",
    "    # new_dual_means += random_pts\n",
    "    new_dual_means = np.array(new_dual_means)\n",
    "    new_dual_means\n",
    "\n",
    "    traj[\"pre_avoid_states\"] = new_dual_means"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "# function for checking whether or not trajectory successfully avoids all avoid regions\n",
    "\n",
    "def check_success(observations, box):\n",
    "    obs = observations[:, :3]\n",
    "    # print(obs)\n",
    "    x1, y1, z1, x2, y2, z2 = box\n",
    "    # print(box)\n",
    "    a = np.array([x1, y1, z1])\n",
    "    b = np.array([x2, y2, z2])\n",
    "    # print(np.count_nonzero(np.all(obs >= a, axis = 1) & np.all(obs <= b, axis = 1)))\n",
    "    return not np.any(np.all(obs >= a, axis = 1) & np.all(obs <= b, axis = 1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "# hindsight relabeling pass 1\n",
    "\n",
    "def add_avoid_states(traj):\n",
    "    observations = traj['observations']\n",
    "    start_pos = observations[0][:3]\n",
    "    end_pos = observations[-1][:3]\n",
    "    if not traj['pre_avoid_states'].shape[0]:\n",
    "        avoid_centroid = start_pos + (end_pos - start_pos) * np.random.rand()\n",
    "    else:\n",
    "        avoid_centroid = traj['pre_avoid_states'][0, :]\n",
    "    random_size = (0.1 + 0.9 * np.random.rand()) * 0.08\n",
    "    center_x, center_y, center_z = avoid_centroid[:3]\n",
    "    # corners = [[center_x + a, center_y + b, center_z + c] for a in displacements for b in displacements for c in displacements]\n",
    "    avoid_box = np.array([center_x - random_size, center_y - random_size, center_z - random_size, center_x + random_size, center_y + random_size, center_z + random_size])\n",
    "    success = check_success(observations, avoid_box)\n",
    "\n",
    "    traj[\"avoid_states\"] = np.array([avoid_box])\n",
    "    traj[\"success\"] = success\n",
    "\n",
    "    return success\n",
    "\n",
    "# hindsight relabeling pass 2\n",
    "\n",
    "def add_avoid_states_alt(traj, paired_success_status):\n",
    "    success = paired_success_status\n",
    "    while success == paired_success_status:\n",
    "        observations = traj['observations']\n",
    "        start_pos = observations[0][:3]\n",
    "        end_pos = observations[-1][:3]\n",
    "        avoid_centroid = start_pos + (end_pos - start_pos) * np.random.rand()\n",
    "        random_size = np.random.rand() * 0.08\n",
    "        center_x, center_y, center_z = avoid_centroid[:3]\n",
    "        # corners = [[center_x + a, center_y + b, center_z + c] for a in displacements for b in displacements for c in displacements]\n",
    "        avoid_box = np.array([center_x - random_size, center_y - random_size, center_z - random_size, center_x + random_size, center_y + random_size, center_z + random_size])\n",
    "        success = check_success(observations, avoid_box)\n",
    "\n",
    "    traj[\"avoid_states\"] = np.array([avoid_box])\n",
    "    traj[\"success\"] = success\n",
    "\n",
    "    return success\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1061/2015748035.py:72: DeprecationWarning: scipy.arccos is deprecated and will be removed in SciPy 2.0.0, use numpy.lib.scimath.arccos instead\n",
      "  angle1 = arccos(np.clip(np.dot(v1_u, v2_u), -1.0, 1.0))\n",
      "/tmp/ipykernel_1061/2015748035.py:73: DeprecationWarning: scipy.arccos is deprecated and will be removed in SciPy 2.0.0, use numpy.lib.scimath.arccos instead\n",
      "  angle2 = arccos(np.clip(np.dot(v3_u, v4_u), -1.0, 1.0))\n"
     ]
    }
   ],
   "source": [
    "successes1, successes2 = [], []\n",
    "for traj, traj2 in zip(data_radt, data_radt_copy):\n",
    "    add_pre_avoid_states(traj)\n",
    "    success1 = add_avoid_states(traj)\n",
    "    successes1.append(success1)\n",
    "    traj[\"observations\"] = traj[\"observations\"][:,:10]\n",
    "\n",
    "    add_pre_avoid_states(traj2)\n",
    "    success2 = add_avoid_states_alt(traj2, success1)\n",
    "    successes2.append(success2)\n",
    "    traj2[\"observations\"] = traj2[\"observations\"][:,:10]\n",
    "\n",
    "print(np.mean(successes1))\n",
    "# print(np.mean(successes2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_radt = data_radt + data_radt_copy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Export dataset to radt dataset directory\n",
    "\n",
    "with open('dataset/reach_obstacle_massless_avoidboxstates4_max0.08.pkl', 'wb') as handle:\n",
    "    # Load the data from the file\n",
    "    pickle.dump(data_radt, handle)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## pointmaze obstacle"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Convert dataset generated for baselines -> RADT format"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "import numpy as np\n",
    "from raDT.constants import *\n",
    "\n",
    "# Open the pickle file for a dataset in the baselines directory in read binary mode\n",
    "with open(HOME_PATH_BASELINES + 'offline_data/random/PointMazeObstacle/buffer_numavoid1_umaze300.pkl', 'rb') as f:\n",
    "    data = pickle.load(f)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(6666, 300, 6)\n",
      "(6666, 300, 2)\n",
      "(6666, 299, 2)\n",
      "(6666, 299, 2)\n",
      "(6666, 299, 1)\n",
      "(6666, 299, 1)\n"
     ]
    }
   ],
   "source": [
    "# Confirm data shapes\n",
    "\n",
    "print(data['o'].shape)\n",
    "print(data['ag'].shape)\n",
    "print(data['g'].shape)\n",
    "print(data['u'].shape)\n",
    "print(data['r'].shape)\n",
    "print(data['c'].shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_radt = []\n",
    "for i in range(data['o'].shape[0]):\n",
    "    traj = {}\n",
    "    traj[\"observations\"] = data['o'][i, :-1, :]\n",
    "    traj['next_observations'] = data['o'][i, 1:, :]\n",
    "    traj['actions'] = data['u'][i, :, :]\n",
    "    traj['timesteps'] = traj[\"observations\"].shape[0]\n",
    "    traj['rewards'] = data['r'][i, :, :].flatten()\n",
    "    traj['costs'] = data['c'][i, :, :].flatten()\n",
    "    traj['terminals'] = np.array([False] * (traj['timesteps'] - 1) + [True])\n",
    "    traj['goal_pos'] = list(data['g'][i, 0])\n",
    "    data_radt = data_radt + [traj]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "\n",
    "# Write dataset to radt dataset directory\n",
    "with open('dataset/pointmaze_obstacle_numavoid1_umaze300.pkl', 'wb') as handle:\n",
    "    pickle.dump(data_radt, handle)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### RADT data prep (hindsight avoid region relabeling)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 117,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "import mpld3\n",
    "# calculate avoid box inference\n",
    "\n",
    "from scipy.spatial import ConvexHull\n",
    "\n",
    "from concave_hull import convex_hull, convex_hull_indexes, concave_hull, concave_hull_indexes\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "  \n",
    "# importing required libraries \n",
    "from mpl_toolkits.mplot3d import Axes3D \n",
    "import matplotlib.pyplot as plt \n",
    "import copy\n",
    "import itertools\n",
    "\n",
    "import numpy as np\n",
    "from scipy import arccos\n",
    "from scipy.linalg import norm\n",
    "\n",
    "# Open the pickle file in read binary mode\n",
    "with open('dataset/pointmaze_obstacle_numavoid1_umaze300.pkl', 'rb') as f:\n",
    "    # Load the data from the file\n",
    "    data_radt = pickle.load(f)\n",
    "\n",
    "data_radt_copy = copy.deepcopy(data_radt)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "metadata": {},
   "outputs": [],
   "source": [
    "def check_success(observations, box):\n",
    "    obs = observations[:, :2]\n",
    "    # print(obs)\n",
    "    x1, y1, x2, y2 = box\n",
    "    # print(box)\n",
    "    a = np.array([x1, y1])\n",
    "    b = np.array([x2, y2])\n",
    "    # print(np.count_nonzero(np.all(obs >= a, axis = 1) & np.all(obs <= b, axis = 1)))\n",
    "    return not np.any(np.all(obs >= a, axis = 1) & np.all(obs <= b, axis = 1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 130,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Hindsight Avoid Region Relabeling Pass 1 \n",
    "\n",
    "def add_avoid_states(traj, n_avoid):\n",
    "    observations = traj['observations']\n",
    "    start_pos = observations[0][:2]\n",
    "    end_pos = observations[-1][:2]\n",
    "\n",
    "    avoid_boxes = []\n",
    "    successes = []\n",
    "    for i in range(n_avoid):\n",
    "        # sampling avoid centroids only from accessible area of state space\n",
    "        avoid_centroid = np.array([np.random.rand() * 3 - 1.5, np.random.rand() * 3 - 1.5]) # generate random point between -1.5 and 1.5\n",
    "        while (-1.5 <= avoid_centroid[0] <= 0.5) and (-0.5 <= avoid_centroid[1] < 0.5):\n",
    "            avoid_centroid = np.array([np.random.rand() * 3 - 1.5, np.random.rand() * 3 - 1.5]) # generate random point between -1.5 and 1.5\n",
    "        random_size = (0.75 + 0.25 * np.random.rand()) * 0.2\n",
    "        center_x, center_y = avoid_centroid[:2]\n",
    "        # corners = [[center_x + a, center_y + b, center_z + c] for a in displacements for b in displacements for c in displacements]\n",
    "        avoid_box = np.array([center_x - random_size, center_y - random_size, center_x + random_size, center_y + random_size])\n",
    "        success = check_success(observations, avoid_box)\n",
    "        avoid_boxes.append(avoid_box)\n",
    "        successes.append(success)\n",
    "\n",
    "    traj[\"avoid_states\"] = np.array(avoid_boxes)\n",
    "    traj[\"success\"] = np.all(successes)\n",
    "\n",
    "    return traj[\"success\"]\n",
    "\n",
    "# Hindsight Avoid Region Relabeling Pass 2\n",
    "\n",
    "def add_avoid_states_alt(traj, paired_success_status, n_avoid):\n",
    "    total_success = paired_success_status\n",
    "    observations = traj['observations']\n",
    "    while total_success == paired_success_status:\n",
    "        avoid_boxes = []\n",
    "        successes = []\n",
    "        for i in range(n_avoid): \n",
    "            # sampling avoid centroids only from accessible area of state space\n",
    "            avoid_centroid = np.array([np.random.rand() * 3 - 1.5, np.random.rand() * 3 - 1.5]) # generate random point between -1.5 and 1.5\n",
    "            while (-1.5 <= avoid_centroid[0] <= 0.5) and (-0.5 <= avoid_centroid[1] < 0.5):\n",
    "                avoid_centroid = np.array([np.random.rand() * 3 - 1.5, np.random.rand() * 3 - 1.5]) # generate random point between -1.5 and 1.5\n",
    "            random_size = (0.75 + 0.25 * np.random.rand()) * 0.2\n",
    "            center_x, center_y = avoid_centroid[:2]\n",
    "            avoid_box = np.array([center_x - random_size, center_y - random_size, center_x + random_size, center_y + random_size])\n",
    "            success = check_success(observations, avoid_box)\n",
    "            avoid_boxes.append(avoid_box)\n",
    "            successes.append(success)\n",
    "        total_success = np.all(successes)\n",
    "    assert total_success != paired_success_status\n",
    "    traj[\"avoid_states\"] = np.array(avoid_boxes)\n",
    "    # print(\"debug:\", traj[\"avoid_states\"].shape)\n",
    "    traj[\"success\"] = total_success\n",
    "\n",
    "    return total_success"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_avoid=3\n",
    "successes1, successes2 = [], []\n",
    "# i = 0\n",
    "for traj, traj2 in zip(data_radt, data_radt_copy):\n",
    "    # if i % 1000 == 0:\n",
    "    #     print(i)\n",
    "    # i+=1\n",
    "    success1 = add_avoid_states(traj, n_avoid)\n",
    "    successes1.append(success1)\n",
    "    traj[\"observations\"] = traj[\"observations\"][:,:4]\n",
    "\n",
    "    success2 = add_avoid_states_alt(traj2, success1, n_avoid)\n",
    "    assert success2 != success1\n",
    "    successes2.append(success2)\n",
    "    traj2[\"observations\"] = traj2[\"observations\"][:,:4]\n",
    "\n",
    "print(np.mean(successes1))\n",
    "print(np.mean(successes2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 132,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_radt = data_radt + data_radt_copy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 143,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Export to radt dataset\n",
    "with open('dataset/pointmaze_obstacle_numavoid3_umaze300_avoidboxstates4_max0.2_newrelabelmethod.pkl', 'wb') as handle:\n",
    "    # Load the data from the file\n",
    "    pickle.dump(data_radt, handle)\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.16"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
