{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "R6DtqWieyKew",
    "outputId": "a4ae8110-a09f-4701-c967-cd4f407871f9"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Collecting rbo\n",
      "  Downloading rbo-0.1.3-py3-none-any.whl.metadata (4.6 kB)\n",
      "Requirement already satisfied: numpy<2.0,>=1.18 in /usr/local/lib/python3.10/dist-packages (from rbo) (1.26.4)\n",
      "Requirement already satisfied: tqdm<5.0.0,>=4.59.0 in /usr/local/lib/python3.10/dist-packages (from rbo) (4.66.5)\n",
      "Downloading rbo-0.1.3-py3-none-any.whl (7.8 kB)\n",
      "Installing collected packages: rbo\n",
      "Successfully installed rbo-0.1.3\n"
     ]
    }
   ],
   "source": [
    "!pip install rbo\n",
    "import rbo\n",
    "import pickle\n",
    "import os\n",
    "import glob\n",
    "import numpy as np\n",
    "from collections import Counter\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "from sklearn.metrics import accuracy_score, classification_report\n",
    "import xgboost as xgb\n",
    "from imblearn.over_sampling import SMOTE, BorderlineSMOTE\n",
    "from sklearn.model_selection import RepeatedStratifiedKFold, GridSearchCV\n",
    "from scipy.stats import kurtosis, skew\n",
    "#from sklearn.metrics import plot_confusion_matrix\n",
    "\n",
    "import seaborn as sns\n",
    "import os\n",
    "import pickle\n",
    "import numpy as np\n",
    "import cv2\n",
    "import keras\n",
    "from keras.applications.imagenet_utils import decode_predictions\n",
    "import skimage.io\n",
    "from skimage.segmentation import quickshift, mark_boundaries\n",
    "from skimage.measure import regionprops\n",
    "import copy\n",
    "import random\n",
    "import sklearn\n",
    "import sklearn.metrics\n",
    "from sklearn.linear_model import LinearRegression\n",
    "from sklearn.linear_model import Ridge\n",
    "from skimage import filters\n",
    "import pandas as pd\n",
    "import warnings\n",
    "import tensorflow as tf\n",
    "import pickle\n",
    "from scipy.stats import kendalltau\n",
    "import sys\n",
    "import scipy.stats as stats\n",
    "from scipy.stats import wilcoxon\n",
    "import itertools\n",
    "from sklearn.ensemble import RandomForestRegressor\n",
    "from sklearn.model_selection import GridSearchCV\n",
    "from sklearn.linear_model import Lasso\n",
    "from functools import partial\n",
    "from sklearn.metrics import confusion_matrix\n",
    "from sklearn.metrics import roc_curve\n",
    "\n",
    "from matplotlib import pyplot as plt\n",
    "import time\n",
    "from sklearn.utils import resample\n",
    "from scipy.stats import norm, gaussian_kde\n",
    "from sklearn.neighbors import KernelDensity\n",
    "import csv\n",
    "\n",
    "import matplotlib.colors as mcolors\n",
    "from skimage.transform import resize\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.linear_model import LinearRegression\n",
    "from sklearn.metrics import mean_squared_error\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.metrics import f1_score, make_scorer\n",
    "\n",
    "from sklearn.metrics import roc_curve, auc\n",
    "from imblearn.pipeline import Pipeline\n",
    "from imblearn.over_sampling import BorderlineSMOTE\n",
    "import shutil\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from imblearn.pipeline import Pipeline\n",
    "from imblearn.over_sampling import BorderlineSMOTE\n",
    "from imblearn.under_sampling import EditedNearestNeighbours\n",
    "import xgboost as xgb\n",
    "from sklearn.model_selection import GridSearchCV, RepeatedStratifiedKFold\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.metrics import roc_curve, auc\n",
    "from scipy.stats import sem\n",
    "from sklearn.metrics import roc_auc_score\n",
    "import gc\n",
    "import matplotlib as mpl"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "4INDVi98yhUr"
   },
   "outputs": [],
   "source": [
    "!cp /content/drive/MyDrive/Results/sanity/datasets/*.* . # get datasets\n",
    "!unzip imagenette_500.zip\n",
    "!unzip pvoc_500.zip"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "DmWSCSQKzIxa",
    "outputId": "d45143ed-3f32-433b-caf0-86bdbcc06c44"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Oxford-IIIT Pets dataset downloaded and extracted successfully.\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import urllib.request\n",
    "import tarfile\n",
    "\n",
    "def download_and_extract(url, extract_dir):\n",
    "    # Create the directory if it doesn't exist\n",
    "    if not os.path.exists(extract_dir):\n",
    "        os.makedirs(extract_dir)\n",
    "\n",
    "    # Download the tar file\n",
    "    tar_filename = os.path.join(extract_dir, \"dataset.tar.gz\")\n",
    "    urllib.request.urlretrieve(url, tar_filename)\n",
    "\n",
    "    # Extract the tar file\n",
    "    with tarfile.open(tar_filename, 'r:gz') as tar:\n",
    "        tar.extractall(path=extract_dir)\n",
    "\n",
    "    # Remove the downloaded tar file\n",
    "    os.remove(tar_filename)\n",
    "\n",
    "# URL to download the dataset\n",
    "url = \"http://www.robots.ox.ac.uk/~vgg/data/pets/data/images.tar.gz\"\n",
    "\n",
    "# Directory to extract the dataset\n",
    "extract_dir = \"./oxpets\"\n",
    "\n",
    "# Download and extract the dataset\n",
    "download_and_extract(url, extract_dir)\n",
    "\n",
    "print(\"Oxford-IIIT Pets dataset downloaded and extracted successfully.\")\n",
    "!rm -rf oxpets_500\n",
    "!mkdir oxpets_500\n",
    "!cp oxpets/images/* oxpets_500"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "ITBCXrob1lQF"
   },
   "outputs": [],
   "source": [
    "# Define the directory path\n",
    "directory = \"oxpets_500/\"\n",
    "\n",
    "# List all image files in the directory (e.g., .jpg, .png)\n",
    "image_files = [f for f in os.listdir(directory) if f.endswith(('.jpg', '.png'))]\n",
    "\n",
    "# Select 50 random files to keep\n",
    "files_to_keep = random.sample(image_files, 500)\n",
    "\n",
    "# Determine which files to delete\n",
    "files_to_delete = set(image_files) - set(files_to_keep)\n",
    "\n",
    "# Delete the files\n",
    "for file_name in files_to_delete:\n",
    "    file_path = os.path.join(directory, file_name)\n",
    "    os.remove(file_path)\n",
    "    print(f\"Deleted {file_name}\")\n",
    "\n",
    "print(\"Done!\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "w5BYDT7pyQQe"
   },
   "outputs": [],
   "source": [
    "def select_n_random_pixels(img, n):\n",
    "    width, height, _ = img.shape\n",
    "    rows, columns, _ = img.shape\n",
    "\n",
    "    pixel_indices = [(y, x) for y in range(height) for x in range(width)]  # Correct order: (row, column)\n",
    "    selected_pixels = random.sample(pixel_indices, n)\n",
    "    return selected_pixels\n",
    "\n",
    "\n",
    "def mse(img1, img2):\n",
    "    h, w, c = img1.shape\n",
    "    diff = cv2.subtract(img1, img2)\n",
    "    err = np.sum(diff ** 2)\n",
    "    mse = err / (float(h * w))\n",
    "    return mse\n",
    "\n",
    "\n",
    "def generate_blur_images(img, coord, sigma, patch_size=9):\n",
    "    height, width, _ = img.shape\n",
    "\n",
    "    y_min = max(int(coord[1] - (patch_size / 2)), 0)\n",
    "    y_max = min(int(coord[1] + (patch_size / 2)), height - 1)\n",
    "\n",
    "    x_min = max(int(coord[0] - (patch_size / 2)), 0)\n",
    "    x_max = min(int(coord[0] + (patch_size / 2)), width - 1)\n",
    "\n",
    "    blur_img = filters.gaussian(img, sigma, channel_axis=2)\n",
    "    blur_img = blur_img * 255  # rescaling to 0 to 255 range from 0 to 1\n",
    "    mask = np.ones(img_size)\n",
    "    mask[x_min:x_max, y_min:y_max] = 0\n",
    "    mask3d = cv2.merge((mask, mask, mask))\n",
    "    # skimage.io.imsave(\"mask_\" + str(imp_level) + \".png\", mask3d)\n",
    "\n",
    "    return perturbed_image\n",
    "\n",
    "\n",
    "def generate_perturbed_images(img, coords, patch_size=9, pert_type=\"T\"):\n",
    "    height, width, _ = img.shape\n",
    "    pert_images = []\n",
    "\n",
    "    for coord in coords:\n",
    "        mask = np.zeros((height, width), dtype=np.uint8)\n",
    "        y_min = max(int(coord[1] - (patch_size / 2)), 0)\n",
    "        y_max = min(int(coord[1] + (patch_size / 2)), height - 1)\n",
    "\n",
    "        x_min = max(int(coord[0] - (patch_size / 2)), 0)\n",
    "        x_max = min(int(coord[0] + (patch_size / 2)), width - 1)\n",
    "\n",
    "        mask[x_min:x_max, y_min:y_max] = 255\n",
    "        damaged_image = copy.deepcopy(img)\n",
    "        damaged_image[x_min:x_max, y_min:y_max] = 0\n",
    "        # damaged_image[coord]=0\n",
    "\n",
    "        if pert_type == 'IT':\n",
    "            output = cv2.inpaint(damaged_image, mask, 5, cv2.INPAINT_TELEA)\n",
    "        elif pert_type == 'IN':\n",
    "            output = cv2.inpaint(damaged_image, mask, 5, cv2.INPAINT_NS)\n",
    "        elif pert_type == 'U0':\n",
    "            output = copy.deepcopy(img)\n",
    "            output[x_min:x_max, y_min:y_max] = [0, 0, 0]\n",
    "        elif pert_type == 'U.5':\n",
    "            output = copy.deepcopy(img)\n",
    "            output[x_min:x_max, y_min:y_max] = [127, 127, 127]\n",
    "        elif pert_type == 'U1':\n",
    "            output = copy.deepcopy(img)\n",
    "            output[x_min:x_max, y_min:y_max] = [255, 255, 255]\n",
    "        elif pert_type == 'FR':\n",
    "            output = copy.deepcopy(img)\n",
    "            random_pixel = [random.randint(0, 255) for _ in range(3)]\n",
    "            output[x_min:x_max, y_min:y_max] = random_pixel\n",
    "        elif pert_type == 'G9':\n",
    "            output = generate_blur_images(img, coord, sigma=0.9)\n",
    "        elif pert_type == 'G15':\n",
    "            output = generate_blur_images(img, coord, sigma=1.5)\n",
    "        else:\n",
    "            print(\"Unknown Type\")\n",
    "            return None\n",
    "        # cv2.line(output, (coord[0], 0), (coord[0], img.shape[0]), (0, 0, 255), 2)\n",
    "        # cv2.line(output, (0, coord[1]), (img.shape[0], coord[1]), (0, 255, 0), 2)\n",
    "\n",
    "        pert_image = copy.deepcopy(img)\n",
    "        pert_image[coord] = output[coord]\n",
    "        pert_image = tf.keras.preprocessing.image.img_to_array(pert_image)\n",
    "        pert_image = preprocess_input(pert_image)\n",
    "        pert_images.append(pert_image)\n",
    "\n",
    "    return pert_images\n",
    "\n",
    "\n",
    "def generate_blur_images_seg(img, mask, sigma):\n",
    "    blur_img = filters.gaussian(img, sigma, channel_axis=2)\n",
    "    blur_img = (blur_img * 255).astype(np.uint8)  # Rescale to 0-255 range\n",
    "    mask3d = np.stack([mask] * 3, axis=-1)\n",
    "    perturbed_image = np.where(mask3d == 255, blur_img, img)\n",
    "\n",
    "    return perturbed_image\n",
    "\n",
    "\n",
    "def generate_perturbed_images_seg(img, segments, seg_ids, pert_type=\"IT\", max_pert_imgs=50):\n",
    "    pert_images = []\n",
    "\n",
    "    for segment_val in seg_ids:\n",
    "        mask = np.zeros(segments.shape, dtype=np.uint8)\n",
    "        mask[segments == segment_val] = 255\n",
    "\n",
    "        damaged_image = copy.deepcopy(img)\n",
    "        damaged_image[segments == segment_val] = 0\n",
    "\n",
    "        if pert_type == 'IT':\n",
    "            output = cv2.inpaint(damaged_image, mask, 5, cv2.INPAINT_TELEA)\n",
    "        elif pert_type == 'IN':\n",
    "            output = cv2.inpaint(damaged_image, mask, 5, cv2.INPAINT_NS)\n",
    "        elif pert_type == 'U0':\n",
    "            output = copy.deepcopy(img)\n",
    "            output[segments == segment_val] = [0, 0, 0]\n",
    "        elif pert_type == 'U.5':\n",
    "            output = copy.deepcopy(img)\n",
    "            output[segments == segment_val] = [127, 127, 127]\n",
    "        elif pert_type == 'U1':\n",
    "            output = copy.deepcopy(img)\n",
    "            output[segments == segment_val] = [255, 255, 255]\n",
    "        elif pert_type == 'FR':\n",
    "            output = copy.deepcopy(img)\n",
    "            random_pixel = [random.randint(0, 255) for _ in range(3)]\n",
    "            output[segments == segment_val] = random_pixel\n",
    "        elif pert_type == 'G3':\n",
    "            output = generate_blur_images_seg(img, mask, sigma=0.3)\n",
    "        elif pert_type == 'G9':\n",
    "            output = generate_blur_images_seg(img, mask, sigma=0.9)\n",
    "        elif pert_type == 'G15':\n",
    "            output = generate_blur_images_seg(img, mask, sigma=1.5)\n",
    "        else:\n",
    "            print(\"Unknown Type\")\n",
    "            return None\n",
    "\n",
    "        pert_image = tf.keras.preprocessing.image.img_to_array(output)\n",
    "        pert_image = preprocess_input(pert_image)\n",
    "        pert_images.append(pert_image)\n",
    "\n",
    "    return pert_images\n",
    "\n",
    "\n",
    "def get_model_params(model_name):\n",
    "    if model_name == \"resnet50\":\n",
    "        model = tf.keras.applications.resnet50.ResNet50(weights='imagenet', include_top=True)\n",
    "        img_size = (224, 224)\n",
    "        preprocess_input = tf.keras.applications.resnet50.preprocess_input\n",
    "\n",
    "    elif model_name == \"inceptionv3\":\n",
    "        model = tf.keras.applications.inception_v3.InceptionV3(weights='imagenet')\n",
    "        img_size = (299, 299)\n",
    "        preprocess_input = tf.keras.applications.inception_v3.preprocess_input\n",
    "\n",
    "    elif model_name == \"xception\":\n",
    "        model = tf.keras.applications.xception.Xception(weights='imagenet')\n",
    "        img_size = (299, 299)\n",
    "        preprocess_input = tf.keras.applications.xception.preprocess_input\n",
    "\n",
    "    else:\n",
    "        print(\"Unknown Model:\", model_name)\n",
    "        return None, None, None\n",
    "\n",
    "    return model, img_size, preprocess_input\n",
    "\n",
    "\n",
    "def get_dataset_params(dataset_name):\n",
    "    if dataset_name == \"oxpets\":\n",
    "        img_dir = \"oxpets_500/\"\n",
    "        result_dir = \"/content/drive/MyDrive/Results/sanity/res_pixel/res_oxpets/\"\n",
    "\n",
    "    elif dataset_name == \"pvoc\":\n",
    "        img_dir = \"pvoc_500/\"\n",
    "        result_dir = \"/content/drive/MyDrive/Results/sanity/res_pixel/res_pvoc/\"\n",
    "\n",
    "    elif dataset_name == \"imagenette\":\n",
    "        img_dir = \"imagenette_500/\"\n",
    "        result_dir = \"/content/drive/MyDrive/Results/sanity/res_pixel/res_imagenette/\"\n",
    "\n",
    "    else:\n",
    "        print(\"Unknown Dataset Name\")\n",
    "        return None, None\n",
    "\n",
    "    return img_dir, result_dir"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "Pk8D7Q9K2r_R"
   },
   "outputs": [],
   "source": [
    "num_pxls = 50\n",
    "batch_size = 10\n",
    "pert_types = [\"IT\", \"IN\", \"U0\", \"U1\", \"U.5\",\"FR\", \"G3\", \"G9\", \"G15\"]\n",
    "\n",
    "model_names = [\"resnet50\", \"inceptionv3\", \"xception\"]\n",
    "datasets = [\"oxpets\", \"imagenette\", \"pvoc\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "ewsmYMdt8Skf",
    "outputId": "bfb3a7f6-65b5-4be5-f4e0-d39cf40e04ad"
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['IT', 'IN', 'U0', 'U1', 'U.5', 'FR', 'G3', 'G9', 'G15']"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pert_types"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "PSiY5x9A27Qy"
   },
   "outputs": [],
   "source": [
    "#segment-wise\n",
    "for dataset in datasets:\n",
    "    img_dir, results_dir = get_dataset_params(dataset)\n",
    "\n",
    "    for model_name in model_names:\n",
    "        model, img_size, preprocess_input = get_model_params(model_name)\n",
    "\n",
    "        modelname = model.name\n",
    "        modelname = model_name.replace(\"_\", \"\")\n",
    "        img_paths = os.listdir(img_dir)\n",
    "        #img_paths = img_paths[0:100] #uncomment to run in test mode\n",
    "        for img_path in img_paths:\n",
    "            print(img_dir + img_path)\n",
    "            img = cv2.imread(img_dir + img_path)\n",
    "\n",
    "            if img is not None and len(img.shape) == 3 and img.shape[2] == 3:\n",
    "                pkl_filename = results_dir + dataset + \"_\" + modelname + \"_\" + img_path.replace('.jpg', '') + '.pkl'\n",
    "                if os.path.exists(pkl_filename):\n",
    "                    continue\n",
    "\n",
    "                img = cv2.resize(img, img_size)\n",
    "                img_arr = tf.keras.preprocessing.image.img_to_array(img)\n",
    "                img_arr = preprocess_input(img_arr)\n",
    "                pred0 = model.predict(np.array([img_arr]), verbose=0)\n",
    "                top_pred_class = pred0[0].argsort()[-1:][::-1]\n",
    "                prob0 = pred0[0][top_pred_class]\n",
    "\n",
    "                # Generate segments using quickshift segmentation\n",
    "                segments = quickshift(img, kernel_size=5, max_dist=6, ratio=0.5)\n",
    "\n",
    "                unique_segments = np.unique(segments)\n",
    "\n",
    "                if len(unique_segments) > num_pxls:\n",
    "                    selected_segments = np.random.choice(unique_segments, num_pxls, replace=False)\n",
    "                else:\n",
    "                    selected_segments = unique_segments\n",
    "\n",
    "                run_dict = {}\n",
    "                for pert_type in pert_types:\n",
    "                    pert_imgs = generate_perturbed_images_seg(img, segments,\n",
    "                        seg_ids= np.unique(selected_segments), pert_type=pert_type)\n",
    "                    probs_arr = model.predict(np.array(pert_imgs), verbose=0)[:, top_pred_class]\n",
    "                    run_dict[pert_type] = np.array(probs_arr)\n",
    "                    del pert_imgs, probs_arr\n",
    "\n",
    "                img_dict = {'run_info': run_dict,\n",
    "                            'prob0': prob0,\n",
    "                            'img_path': dataset + \"/\" + img_path}\n",
    "\n",
    "                print(pkl_filename)\n",
    "                with open(pkl_filename, 'wb') as f1:\n",
    "                    pickle.dump(img_dict, f1)\n",
    "            else:\n",
    "                print(\"Bad Image:\" + str(img_path))\n",
    "                continue"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "m7DqtsfGEPVZ"
   },
   "outputs": [],
   "source": [
    "#pixel-wise\n",
    "for dataset in datasets:\n",
    "    img_dir, results_dir = get_dataset_params(dataset)\n",
    "\n",
    "    for model_name in model_names:\n",
    "        model, img_size, preprocess_input = get_model_params(model_name)\n",
    "\n",
    "        modelname = model.name\n",
    "        modelname = model_name.replace(\"_\", \"\")\n",
    "        img_paths = os.listdir(img_dir)\n",
    "        #img_paths = img_paths[0:100] #uncomment to run in test mode\n",
    "        for img_path in img_paths:\n",
    "            print(img_dir + img_path)\n",
    "            img = cv2.imread(img_dir + img_path)\n",
    "\n",
    "            if img is not None and len(img.shape) == 3 and img.shape[2] == 3:\n",
    "                pkl_filename = results_dir + dataset + \"_\" + modelname + \"_\" + img_path.replace('.jpg', '') + '.pkl'\n",
    "                if os.path.exists(pkl_filename):\n",
    "                    continue\n",
    "\n",
    "                img = cv2.resize(img, img_size)\n",
    "                img_arr = tf.keras.preprocessing.image.img_to_array(img)\n",
    "                img_arr = preprocess_input(img_arr)\n",
    "                pred0 = model.predict(np.array([img_arr]), verbose=0)\n",
    "                top_pred_class = pred0[0].argsort()[-1:][::-1]\n",
    "                prob0 = pred0[0][top_pred_class]\n",
    "\n",
    "                # Generate segments using quickshift segmentation\n",
    "                segments = quickshift(img, kernel_size=5, max_dist=6, ratio=0.5)\n",
    "\n",
    "                sel_pxls = select_n_random_pixels(img, num_pxls)\n",
    "\n",
    "                run_dict = {}\n",
    "                for pert_type in pert_types:\n",
    "                    pert_imgs = generate_perturbed_images(img, sel_pxls, patch_size=9, pert_type=pert_type)\n",
    "                    probs_arr = model.predict(np.array(pert_imgs), verbose=0)[:, top_pred_class]\n",
    "                    run_dict[pert_type] = np.array(probs_arr)\n",
    "                    del pert_imgs, probs_arr\n",
    "\n",
    "                img_dict = {'run_info': run_dict,\n",
    "                            'prob0': prob0,\n",
    "                            'img_path': dataset + \"/\" + img_path}\n",
    "\n",
    "                print(pkl_filename)\n",
    "                with open(pkl_filename, 'wb') as f1:\n",
    "                    pickle.dump(img_dict, f1)\n",
    "            else:\n",
    "                print(\"Bad Image:\" + str(img_path))\n",
    "                continue"
   ]
  }
 ],
 "metadata": {
  "accelerator": "GPU",
  "colab": {
   "gpuType": "A100",
   "machine_shape": "hm",
   "provenance": []
  },
  "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.10.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
