{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 75,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sicore\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "from sklearn import metrics\n",
    "from sklearn.preprocessing import LabelBinarizer\n",
    "import collections\n",
    "from math import gamma\n",
    "from math import pi\n",
    "from tqdm import tqdm\n",
    "from concurrent.futures import ProcessPoolExecutor\n",
    "import scipy\n",
    "import random\n",
    "import csv\n",
    "import itertools\n",
    "import math\n",
    "from sklearn import preprocessing\n",
    "from sklearn.model_selection import train_test_split\n",
    "import os\n",
    "from torchvision import datasets, transforms\n",
    "from PIL import Image\n",
    "from torchvision.utils import make_grid\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from scipy.spatial import distance\n",
    "import cv2\n",
    "from glob import glob"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 76,
   "metadata": {},
   "outputs": [],
   "source": [
    "########################\n",
    "# Setting path\n",
    "########################\n",
    "train_good_path = '/your/path/to/MVtec/bottle/train/good'\n",
    "evaluation_data_path = '/your/path/to/MVtec/bottle/test/broken_large'\n",
    "ground_truth_path = '/your/path/to/MVtec/bottle/ground_truth/broken_large'\n",
    "reference_features_dir = './reference_features'\n",
    "evaluation_features_dir = './evaluation_features'\n",
    "os.makedirs(reference_features_dir, exist_ok=True)\n",
    "os.makedirs(evaluation_features_dir, exist_ok=True)\n",
    "\n",
    "k = 1\n",
    "abnormal_classes = {'broken_large', 'broken_small', 'contamination'}\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "生成された異常パッチの数: 10\n",
      "true_y_list のサイズ: 10\n",
      "異常データの割合: 100.00%\n"
     ]
    }
   ],
   "source": [
    "#Setting parameters\n",
    "n = 100  # training size\n",
    "patch_size = 30  #patch size\n",
    "anomaly_threshold = 0.2  # 20% or more abnormal pixels are defined as anomaly patch\n",
    "num_abnormal_patches = 10  # anomaly patch size\n",
    "test_datasets = []  # test data\n",
    "true_y_list = []  # true label\n",
    "\n",
    "'\n",
    "\n",
    "# Image loading functions\n",
    "def load_images_from_folder(folder):\n",
    "    \"\"\" Load all images from a given folder  \"\"\"\n",
    "    image_paths = sorted(glob(os.path.join(folder, \"*.png\")))\n",
    "    if len(image_paths) == 0:\n",
    "        raise ValueError(f\"⚠️ Error: {folder} has nothing！\")\n",
    "    \n",
    "    images = [cv2.imread(img_path, cv2.IMREAD_GRAYSCALE) for img_path in image_paths]\n",
    "    return np.array(images), image_paths\n",
    "\n",
    "\n",
    "train_images, _ = load_images_from_folder(train_good_path)\n",
    "train_images = random.sample(list(train_images), n)  \n",
    "\n",
    "\n",
    "ground_truth_images, _ = load_images_from_folder(ground_truth_path)\n",
    "\n",
    "\n",
    "while len(test_datasets) < num_abnormal_patches:\n",
    "    \n",
    "    evaluation_images, _ = load_images_from_folder(evaluation_data_path)\n",
    "    evaluation_image = random.choice(evaluation_images)  \n",
    "\n",
    "    \n",
    "    h, w = evaluation_image.shape\n",
    "    top_left_x = random.randint(0, w - patch_size)\n",
    "    top_left_y = random.randint(0, h - patch_size)\n",
    "\n",
    "    \n",
    "    evaluation_patch = evaluation_image[top_left_y:top_left_y + patch_size, top_left_x:top_left_x + patch_size]\n",
    "\n",
    "    \n",
    "    train_patches = np.array([\n",
    "        img[top_left_y:top_left_y + patch_size, top_left_x:top_left_x + patch_size] for img in train_images\n",
    "    ])\n",
    "\n",
    "    \n",
    "    ground_truth_image = random.choice(ground_truth_images)  \n",
    "    anomaly_patch = ground_truth_image[top_left_y:top_left_y + patch_size, top_left_x:top_left_x + patch_size]\n",
    "\n",
    "    \n",
    "    anomaly_ratio = np.sum(anomaly_patch == 255) / (patch_size * patch_size)\n",
    "\n",
    "    \n",
    "    if anomaly_ratio >= anomaly_threshold:\n",
    "        true_y_list.append(1)  \n",
    "        test_datasets.append(np.vstack([evaluation_patch.reshape(1, patch_size, patch_size), train_patches]))  # (n+1, 30, 30)\n",
    "    \n",
    "    \n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 78,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "標準化後の訓練データの形状: (500, 30, 30)\n",
      "標準化後のテストデータの形状: (10, 101, 30, 30)\n",
      "標準化後の全体平均: 93.81619\n",
      "標準化後の全体標準偏差: 63.06458\n"
     ]
    }
   ],
   "source": [
    "train_images_array = np.array(train_images)  # (n, H, W) = (100, 256, 256)\n",
    "test_datasets_array = np.array(test_datasets)  # (500, n+1, 30, 30)\n",
    "\n",
    "\n",
    "all_train_patches = []\n",
    "for img in train_images_array:\n",
    "    for _ in range(5):  \n",
    "        x = np.random.randint(0, img.shape[1] - patch_size)\n",
    "        y = np.random.randint(0, img.shape[0] - patch_size)\n",
    "        all_train_patches.append(img[y:y+patch_size, x:x+patch_size])\n",
    "all_train_patches = np.array(all_train_patches)  # (500, 30, 30)\n",
    "\n",
    "\n",
    "all_patches = np.concatenate([test_datasets_array.reshape(-1, patch_size, patch_size), \n",
    "                              all_train_patches], axis=0)  # (all patch, 30, 30)\n",
    "\n",
    "\n",
    "mean_pixel = np.mean(all_patches)  \n",
    "std_pixel = np.std(all_patches)  \n",
    "\n",
    "\n",
    "std_pixel = np.where(std_pixel == 0, 1e-8, std_pixel)\n",
    "\n",
    "\n",
    "train_images_normalized = (all_train_patches - mean_pixel) / std_pixel  # (500, 30, 30)\n",
    "test_datasets_normalized = (test_datasets_array - mean_pixel) / std_pixel  # (500, n+1, 30, 30)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "metadata": {},
   "outputs": [],
   "source": [
    "iteration = num_abnormal_patches"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "metadata": {},
   "outputs": [],
   "source": [
    "def f(X):\n",
    "    '''\n",
    "    X ... 観測点の行列\n",
    "    検定対象は0番目の点で固定している\n",
    "    '''\n",
    "    distances = []\n",
    "    for p in X[:, 0:M]:\n",
    "        test_X = X[0, 0:M]\n",
    "        distances.append(np.linalg.norm(test_X - p, ord=1))\n",
    "    distances.sort()\n",
    "    ip = distances[k]\n",
    "\n",
    "    #異常度を定義\n",
    "    a = (-1) * np.log(k) + M * np.log(ip)\n",
    "\n",
    "    if a > a_th:\n",
    "        res = 1\n",
    "    else:\n",
    "        res = 0\n",
    "    \n",
    "    return res, a, ip"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "metadata": {},
   "outputs": [],
   "source": [
    "generator = np.random.default_rng()\n",
    "rnd = generator.normal(loc=0, scale=0.01, size=303)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "XX_mat の形状: (10, 90901)\n"
     ]
    }
   ],
   "source": [
    "num_patches_per_test = n + 1  \n",
    "num_features_per_patch = patch_size * patch_size  \n",
    "\n",
    "\n",
    "XX_mat = []\n",
    "\n",
    "for i in range(num_abnormal_patches):\n",
    "    \n",
    "    patch_set = test_datasets_normalized[i]  # (n+1, 30, 30)\n",
    "    \n",
    "\n",
    "    patch_vector = patch_set.reshape(num_patches_per_test * num_features_per_patch)  # (n+1)*900\n",
    "    \n",
    "\n",
    "    labeled_patch_vector = np.append(patch_vector, true_y_list[i])  \n",
    "    \n",
    "    \n",
    "    XX_mat.append(labeled_patch_vector)\n",
    "\n",
    "\n",
    "XX_mat = np.array(XX_mat)  # (num_abnormal_patches, (n+1)*900 + 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 83,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Small size AlexNet\n",
    "class SimpleAlexNet(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(SimpleAlexNet, self).__init__()\n",
    "        self.conv1 = nn.Conv2d(1, 32, kernel_size=5, stride=1, padding=2)  \n",
    "        self.pool = nn.MaxPool2d(kernel_size=2, stride=2)\n",
    "        self.fc = nn.Linear(32 * 15 * 15, 10)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = F.relu(self.conv1(x))\n",
    "        x = self.pool(x)\n",
    "        x = x.view(x.size(0), -1)\n",
    "        x = self.fc(x)\n",
    "        return x\n",
    "\n",
    "\n",
    "model = SimpleAlexNet()\n",
    "model.eval()\n",
    "\n",
    "num_samples = num_abnormal_patches  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 84,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "lat_XX_mat の形状: (10, 727201)\n"
     ]
    }
   ],
   "source": [
    "\n",
    "num_patches_per_test = n + 1  \n",
    "num_features_per_patch = 32 * 15 * 15  \n",
    "num_total_features = num_patches_per_test * num_features_per_patch  # (n+1) * 7200\n",
    "\n",
    "\n",
    "lat_XX_mat = []\n",
    "\n",
    "\n",
    "for i in range(num_abnormal_patches):\n",
    "    \n",
    "    patch_set = test_datasets_normalized[i]  # (n+1, 30, 30)\n",
    "\n",
    "    \n",
    "    patch_tensor = torch.tensor(patch_set, dtype=torch.float32).unsqueeze(1)  # (n+1, 1, 30, 30)\n",
    "\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        feature_map = model.conv1(patch_tensor)  # (n+1, 32, 30, 30)\n",
    "        feature_map = model.pool(feature_map)  # (n+1, 32, 15, 15)\n",
    "        feature_vector = feature_map.view(num_patches_per_test, -1).numpy()  # (n+1, 7200)\n",
    "\n",
    "\n",
    "    feature_vector_row = feature_vector.flatten()  # (num_total_features,)\n",
    "\n",
    "\n",
    "    labeled_feature_vector = np.append(feature_vector_row, true_y_list[i])  \n",
    "\n",
    "    \n",
    "    lat_XX_mat.append(labeled_feature_vector)\n",
    "\n",
    "\n",
    "lat_XX_mat = np.array(lat_XX_mat)  # (num_abnormal_patches, (n+1)*7200 + 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "M = lat_XX_mat.shape[1] - 1\n",
    "k = 1      # k Nearest Neighbors\n",
    "\n",
    "\n",
    "# AD function\n",
    "def f(X):\n",
    "    \n",
    "    distances = []\n",
    "    for p in X[:, 0:M]:  \n",
    "        test_X = X[0, 0:M]  \n",
    "        distances.append(np.linalg.norm(test_X - p, ord=1))\n",
    "    \n",
    "    distances.sort()\n",
    "    ip = distances[k]  \n",
    "\n",
    "\n",
    "    a = (-1) * np.log(k) + M * np.log(ip)\n",
    "\n",
    "    if a > a_th:\n",
    "        res = 1  \n",
    "    else:\n",
    "        res = 0  \n",
    "    \n",
    "    return res, a, ip"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAnYAAAHWCAYAAAD6oMSKAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy81sbWrAAAACXBIWXMAAA9hAAAPYQGoP6dpAABGIUlEQVR4nO3deXxOZ/7/8fctyy1JkyBks4TaSW3VsRSx72ptq9qiaKmlDKat6UzFjFraoTrTVk2rwbSWLnS0Whq7jmjtVVpUSRRpNJZEECLX7w/f3D+3LOKWyO14PR+P83j0XOc61/lcR8S7Z7lvmzHGCAAAAHe8YkVdAAAAAAoGwQ4AAMAiCHYAAAAWQbADAACwCIIdAACARRDsAAAALIJgBwAAYBEEOwAAAIsg2AEAAFgEwQ4oIPPmzZPNZnMsxYsXV2hoqFq1aqWpU6cqKSkp2z7R0dGy2Ww3dZzz588rOjpa69evv6n9cjpWxYoV1bVr15sa50YWLlyoWbNm5bjNZrMpOjq6QI9X0NasWaOGDRvKz89PNptNn3322Q332bNnj2w2m7y8vHTixInCL/I2GzhwoCpWrFhg412+fFlz5szRAw88oFKlSsnX11cRERHq3r27li1bVmDHAe5GBDuggMXExCguLk6xsbF66623VK9ePU2fPl01a9bU6tWrnfoOGTJEcXFxNzX++fPnNWnSpJsOdq4cyxV5Bbu4uDgNGTKk0GtwlTFGjzzyiLy8vLR8+XLFxcUpKirqhvu99957kqSMjAwtWLCgsMu84z355JMaNWqUWrVqpQ8++ECff/65/vKXv8jT01OrVq0q6vKAO5pnURcAWE1kZKQaNmzoWO/du7f++Mc/qlmzZurVq5cOHjyokJAQSVK5cuVUrly5Qq3n/Pnz8vX1vS3HupHGjRsX6fFv5Pjx4zp16pR69uypNm3a5Guf9PR0ffjhh6pbt65+//13vf/++3rhhRcKudI71+HDh7VkyRK9/PLLmjRpkqO9TZs2evrpp5WZmXnbajHG6OLFi/Lx8bltxwQKG1fsgNugQoUKmjFjhlJTUzVnzhxHe063R9euXauWLVsqKChIPj4+qlChgnr37q3z58/ryJEjKlOmjCRp0qRJjtu+AwcOdBpvx44d6tOnj0qWLKnKlSvneqwsy5YtU506dVS8eHHde++9+uc//+m0Pes285EjR5za169fL5vN5rh62LJlS61YsULx8fFOt6Wz5HQr9ocfflD37t1VsmRJFS9eXPXq1dP8+fNzPM6iRYv00ksvKTw8XAEBAWrbtq3279+f+4m/xjfffKM2bdrI399fvr6+atq0qVasWOHYHh0d7Qi+L7zwgmw2W75uP3722WdKTk7WkCFDNGDAAB04cEDffPNNtn5Zt71XrlypBg0ayMfHRzVq1ND777+fre/NnJOFCxfqhRdeUFhYmO655x5169ZNv/32m1JTU/XMM8+odOnSKl26tJ566imdO3fOaYy33npLLVq0UHBwsPz8/HTffffp1Vdf1eXLl/Occ5s2bVSjRg0ZY5zajTGqUqWKunTpkuu+ycnJkqSwsLActxcr5vzP0pkzZzRu3Djde++9stvtCg4OVufOnfXTTz85+pw6dUrDhw9X2bJl5e3trXvvvVcvvfSS0tPTncay2WwaOXKk3nnnHdWsWVN2u91xXg8ePKh+/fopODhYdrtdNWvW1FtvveW0f2ZmpiZPnqzq1avLx8dHJUqUUJ06dfTGG2/keb6A24krdsBt0rlzZ3l4eGjjxo259jly5Ii6dOmi5s2b6/3331eJEiV07NgxrVy5UpcuXVJYWJhWrlypjh07avDgwY7bmllhL0uvXr3Ut29fDRs2TGlpaXnWtWvXLo0ZM0bR0dEKDQ3Vhx9+qNGjR+vSpUsaP378Tc3x7bff1jPPPKNDhw7l61mp/fv3q2nTpgoODtY///lPBQUF6YMPPtDAgQP122+/6fnnn3fq/+c//1kPPvig3nvvPaWkpOiFF15Qt27d9OOPP8rDwyPX42zYsEHt2rVTnTp1NHfuXNntdr399tvq1q2bFi1apEcffVRDhgxR3bp11atXL40aNUr9+vWT3W6/4Ryyxnv88cd16tQpTZ06VXPnzlWzZs2y9d29e7fGjRunF198USEhIXrvvfc0ePBgValSRS1atHD5nLRq1Urz5s3TkSNHNH78eD322GPy9PRU3bp1tWjRIu3cuVN//vOf5e/v7xTaDx06pH79+qlSpUry9vbW7t279corr+inn37KMXBmGT16tLp37641a9aobdu2jvavvvpKhw4dyvY/BteqWbOmSpQooUmTJqlYsWJq3759rgE6NTVVzZo105EjR/TCCy+oUaNGOnfunDZu3KgTJ06oRo0aunjxolq1aqVDhw5p0qRJqlOnjjZt2qSpU6dq165dTuFduhrEN23apJdfflmhoaEKDg7Wvn371LRpU8f/gIWGhmrVqlV67rnn9Pvvv2vixImSpFdffVXR0dH6y1/+ohYtWujy5cv66aefdObMmVznC9x2BkCBiImJMZLM1q1bc+0TEhJiatas6VifOHGiufav4SeffGIkmV27duU6xsmTJ40kM3HixGzbssZ7+eWXc912rYiICGOz2bIdr127diYgIMCkpaU5ze3w4cNO/datW2ckmXXr1jnaunTpYiIiInKs/fq6+/bta+x2u0lISHDq16lTJ+Pr62vOnDnjdJzOnTs79fvoo4+MJBMXF5fj8bI0btzYBAcHm9TUVEdbRkaGiYyMNOXKlTOZmZnGGGMOHz5sJJnXXnstz/GyHDlyxBQrVsz07dvX0RYVFWX8/PxMSkqKU9+IiAhTvHhxEx8f72i7cOGCKVWqlBk6dKij7WbPSbdu3Zz6jRkzxkgyzz33nFN7jx49TKlSpXKdy5UrV8zly5fNggULjIeHhzl16pRj24ABA5z+TK9cuWLuvfde071792w1Vq5c2XE+c7NixQpTunRpI8lIMkFBQebhhx82y5cvd+r3t7/9zUgysbGxuY71zjvvGEnmo48+cmqfPn26kWS+/vprR5skExgY6DQ3Y4zp0KGDKVeunDl79qxT+8iRI03x4sUd/bt27Wrq1auX59yAosatWOA2MtfdurpevXr15O3trWeeeUbz58/XL7/84tJxevfune++tWvXVt26dZ3a+vXrp5SUFO3YscOl4+fX2rVr1aZNG5UvX96pfeDAgTp//ny2lz0eeughp/U6depIkuLj43M9Rlpamr799lv16dNH99xzj6Pdw8NDTz75pH799dd83869XkxMjDIzMzVo0CBH26BBg5SWlqYlS5Zk61+vXj1VqFDBsV68eHFVq1bNqf6bPSfXv9Vcs2ZNScp2O7RmzZo6deqU0+3YnTt36qGHHlJQUJA8PDzk5eWl/v3768qVKzpw4ECu8y5WrJhGjhypL774QgkJCZKuXv1buXKlhg8ffsM3vTt37qyEhAQtW7ZM48ePV+3atfXZZ5/poYce0siRIx39vvrqK1WrVs3pquD11q5dKz8/P/Xp08epPevxhDVr1ji1t27dWiVLlnSsX7x4UWvWrFHPnj3l6+urjIwM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R5WcnKy//e1vOnHihCIjI/Xll18qIiKiqEsDAABwK24f7CRp+PDhGj58eFGXkSe73a6JEydmuw18N2DuzP1uczfPXbq758/cmbu7sxmTn3dnAQAA4O7c+gOKAQAAkH8EOwAAAIsg2AEAAFjEXR3sXnnlFTVt2lS+vr65ftZdQkKCunXrJj8/P5UuXVrPPfecLl265NRnz549ioqKko+Pj8qWLau//e1v2b72Y8OGDbr//vtVvHhx3XvvvXrnnXeyHevTTz9VrVq1ZLfbVatWLS1btixbn7fffluVKlVS8eLFdf/992vTpk2un4BrHDhwQN27d1fp0qUVEBCgBx98UOvWrXPq427noqCtWLFCjRo1ko+Pj0qXLq1evXo5bbf6/NPT01WvXj3ZbDbt2rXLaZsV537kyBENHjxYlSpVko+PjypXrqyJEydmm5cV534rCut3UGGZOnWqHnjgAfn7+ys4OFg9evTQ/v37nfoYYxQdHa3w8HD5+PioZcuW2rt3r1Of9PR0jRo1SqVLl5afn58eeugh/frrr059Tp8+rSeffFKBgYEKDAzUk08+qTNnzjj1yc/PU2GZOnWqbDabxowZ42iz8tyPHTumJ554QkFBQfL19VW9evW0fft268/d9S/8uvO9/PLLZubMmWbs2LEmMDAw2/aMjAwTGRlpWrVqZXbs2GFiY2NNeHi4GTlypKPP2bNnTUhIiOnbt6/Zs2eP+fTTT42/v7/5xz/+4ejzyy+/GF9fXzN69Gizb98+8+677xovLy/zySefOPps3rzZeHh4mClTppgff/zRTJkyxXh6epotW7Y4+ixevNh4eXmZd9991+zbt8+MHj3a+Pn5mfj4+Fs+F1WqVDGdO3c2u3fvNgcOHDDDhw83vr6+5sSJE255LgraJ598YkqWLGlmz55t9u/fb3766Sfz8ccfO7Zbff7GGPPcc8+ZTp06GUlm586dlp/7V199ZQYOHGhWrVplDh06ZP773/+a4OBgM27cOMvP3VWF+TuosHTo0MHExMSYH374wezatct06dLFVKhQwZw7d87RZ9q0acbf3998+umnZs+ePebRRx81YWFhJiUlxdFn2LBhpmzZsiY2Ntbs2LHDtGrVytStW9dkZGQ4+nTs2NFERkaazZs3m82bN5vIyEjTtWtXx/b8/DwVlu+++85UrFjR1KlTx4wePdrycz916pSJiIgwAwcONN9++605fPiwWb16tfn5558tP/e7OthliYmJyTHYffnll6ZYsWLm2LFjjrZFixYZu91uzp49a4wx5u233zaBgYHm4sWLjj5Tp0414eHhJjMz0xhjzPPPP29q1KjhNPbQoUNN48aNHeuPPPKI6dixo1OfDh06mL59+zrW//CHP5hhw4Y59alRo4Z58cUXb3LGzk6ePGkkmY0bNzraUlJSjCSzevVqY4z7nYuCdPnyZVO2bFnz3nvv5drHyvM35ur8atSoYfbu3Zst2Fl97td69dVXTaVKlRzrd9Pc86OwfgfdTklJSUaS2bBhgzHGmMzMTBMaGmqmTZvm6HPx4kUTGBho3nnnHWOMMWfOnDFeXl5m8eLFjj7Hjh0zxYoVMytXrjTGGLNv3z4jySmIx8XFGUnmp59+Msbk7+epMKSmppqqVaua2NhYExUV5Qh2Vp77Cy+8YJo1a5brdivP/a6+FXsjcXFxioyMdPputg4dOig9Pd1xOTcuLk5RUVFOn23ToUMHHT9+XEeOHHH0ad++vdPYHTp00LZt23T58uU8+2zevFmSdOnSJW3fvj1bn/bt2zv6uCooKEg1a9bUggULlJaWpoyMDM2ZM0chISG6//773e5cFLQdO3bo2LFjKlasmOrXr6+wsDB16tTJ6ZK8lef/22+/6emnn9Z//vMf+fr6Zttu5blf7+zZsypVqpRj/W6a+40U5u+g2+ns2bOS5PhzPnz4sBITE53mZbfbFRUV5ZjX9u3bdfnyZac+4eHhioyMdPSJi4tTYGCgGjVq5OjTuHFjBQYGOvW50c9TYRgxYoS6dOmitm3bOrVbee7Lly9Xw4YN9fDDDys4OFj169fXu+++e1fMnWCXh8TERIWEhDi1lSxZUt7e3kpMTMy1T9b6jfpkZGTo999/z7NP1hi///67rly5kmcfV9lsNsXGxmrnzp3y9/dX8eLF9frrr2vlypWOZw/d6VwUtF9++UWSFB0drb/85S/64osvVLJkSUVFRenUqVO51mSF+RtjNHDgQA0bNkwNGzbMsY9V5369Q4cO6V//+pfT1xXeLXPPj8L8HXS7GGM0duxYNWvWTJGRkZL+/59RXvNKTEyUt7e3SpYsmWef4ODgbMcMDg7O8+fg+p+ngrZ48WLt2LFDU6dOzbbNynP/5ZdfNHv2bFWtWlWrVq3SsGHD9Nxzz2nBggWOerLmkde87sS5Wy7YRUdHy2az5bls27Yt3+PZbLZsbcYYp/br+5j/e2i6IPpc35afPlnyey6MMRo+fLiCg4O1adMmfffdd+revbu6du2qEydOuO25uJH8zj8zM1OS9NJLL6l37966//77FRMTI5vNpo8//viOnH9+5/6vf/1LKSkpmjBhQp7jWXHu1zp+/Lg6duyohx9+WEOGDHHadifN/Xa4E2rMzciRI/X9999r0aJF2ba5Mq8b/Ry42qegHD16VKNHj9YHH3yg4sWL59rPinPPzMxUgwYNNGXKFNWvX19Dhw7V008/rdmzZzv1s+Lc74ivFLsZI0eOVN++ffPsU7FixXyNFRoaqm+//dap7fTp07p8+bIjfYeGhmZL3ElJSZJ0wz6enp4KCgrKs0/WGKVLl5aHh0eefa6X33Oxdu1affHFFzp9+rQCAgIkXX3zLTY2VvPnz9eLL77oVuciv/I7/9TUVElSrVq1HO12u1333nuvEhISHDXdSfPP79wnT56sLVu2ZPuanIYNG+rxxx/X/PnzLTv3LMePH1erVq3UpEkT/fvf/3bqd6fNvTC58jvInYwaNUrLly/Xxo0bVa5cOUd7aGiopKtXVcLCwhzt184rNDRUly5d0unTp52u3iQlJalp06aOPr/99lu24548edJpnBv9PBWk7du3KykpyfFIjSRduXJFGzdu1Jtvvul4O9iKcw8LC3P6nS5JNWvW1KeffuqoR7Lm3Hl5wtz45Ynjx4872hYvXpztwekSJUqY9PR0R59p06Zle3C6Zs2aTmMPGzYs24PTnTp1curTsWPHbC9PPPvss059atasecsPLi9fvtwUK1bMpKamOrVXq1bNvPLKK8YY9zsXBens2bPGbrc7vTxx6dIlExwcbObMmWOMse784+PjzZ49exzLqlWrjCTzySefmKNHjxpjrDt3Y4z59ddfTdWqVU3fvn2d3nLLYuW5u6KwfgcVpszMTDNixAgTHh5uDhw4kOP20NBQM336dEdbenp6jg/RL1myxNHn+PHjOT5E/+233zr6bNmyJceH6PP6eSpIKSkpTn+/9+zZYxo2bGieeOIJs2fPHkvP/bHHHsv28sSYMWNMkyZNjDHW/nO/q4NdfHy82blzp5k0aZK55557zM6dO83OnTsdASfrFeU2bdqYHTt2mNWrV5ty5co5vaJ85swZExISYh577DGzZ88es3TpUhMQEJDjRx388Y9/NPv27TNz587N9lEH//vf/4yHh4eZNm2a+fHHH820adNy/biTuXPnmn379pkxY8YYPz8/c+TIkVs6DydPnjRBQUGmV69eZteuXWb//v1m/PjxxsvLy+zatcstz0VBGz16tClbtqxZtWqV+emnn8zgwYNNcHCwOXXq1F0x/yyHDx/O9eNOrDb3Y8eOmSpVqpjWrVubX3/91Zw4ccKxWH3uriqs30GF6dlnnzWBgYFm/fr1Tn/G58+fd/SZNm2aCQwMNEuXLjV79uwxjz32WI4fe1GuXDmzevVqs2PHDtO6descP/aiTp06Ji4uzsTFxZn77rsvx4+9yOvnqbBd+1asMdad+3fffWc8PT3NK6+8Yg4ePGg+/PBD4+vraz744APLz/2uDnYDBgwwkrIt69atc/SJj483Xbp0MT4+PqZUqVJm5MiRTh9rYIwx33//vWnevLmx2+0mNDTUREdHO/5PPcv69etN/fr1jbe3t6lYsaKZPXt2tno+/vhjU716dePl5WVq1KhhPv3002x93nrrLRMREWG8vb1NgwYNHK/s36qtW7ea9u3bm1KlShl/f3/TuHFj8+WXXzr1cbdzUZAuXbpkxo0bZ4KDg42/v79p27at+eGHH5z6WHn+WXIKdsZYc+4xMTE5/v2//kaGFed+Kwrrd1Bhye3POCYmxtEnMzPTTJw40YSGhhq73W5atGhh9uzZ4zTOhQsXzMiRI02pUqWMj4+P6dq1q0lISHDqk5ycbB5//HHj7+9v/P39zeOPP25Onz7t1Cc/P0+F6fpgZ+W5f/755yYyMtLY7XZTo0YN8+9//9tpu1XnbjPmuo9HBwAAwB3Jcm/FAgAA3K0IdgAAABZBsAMAALAIgh0AAIBFEOwAAAAsgmAHAABgEQQ7AAAAiyDYAQAAWATBDgBcVLFiRc2aNauoywAAB4IdgCK1efNmeXh4qGPHjkVdSpGYM2eO6tatKz8/P5UoUUL169fX9OnTi7osAHcoz6IuAMDd7f3339eoUaP03nvvKSEhQRUqVCjqkm6buXPnauzYsfrnP/+pqKgopaen6/vvv9e+ffsK7ZiXL1+Wl5dXoY0PoGhxxQ5AkUlLS9NHH32kZ599Vl27dtW8efOctq9fv142m01r1qxRw4YN5evrq6ZNm2r//v1O/WbPnq3KlSvL29tb1atX13/+8x+n7TabTXPmzFHXrl3l6+urmjVrKi4uTj///LNatmwpPz8/NWnSRIcOHXLsc+jQIXXv3l0hISG655579MADD2j16tW5zmXQoEHq2rWrU1tGRoZCQ0P1/vvv57jP559/rkceeUSDBw9WlSpVVLt2bT322GP6+9//7tTv/fffV+3atWW32xUWFqaRI0c6tiUkJKh79+665557FBAQoEceeUS//fabY3t0dLTq1aun999/X/fee6/sdruMMTp79qyeeeYZBQcHKyAgQK1bt9bu3btznR+AOwPBDkCRWbJkiapXr67q1avriSeeUExMjIwx2fq99NJLmjFjhrZt2yZPT08NGjTIsW3ZsmUaPXq0xo0bpx9++EFDhw7VU089pXXr1jmN8fe//139+/fXrl27VKNGDfXr109Dhw7VhAkTtG3bNklyCkznzp1T586dtXr1au3cuVMdOnRQt27dlJCQkONchgwZopUrV+rEiROOti+//FLnzp3TI488kuM+oaGh2rJli+Lj43M9R7Nnz9aIESP0zDPPaM+ePVq+fLmqVKkiSTLGqEePHjp16pQ2bNig2NhYHTp0SI8++qjTGD///LM++ugjffrpp9q1a5ckqUuXLkpMTNSXX36p7du3q0GDBmrTpo1OnTqVay0A7gAGAIpI06ZNzaxZs4wxxly+fNmULl3axMbGOravW7fOSDKrV692tK1YscJIMhcuXHCM8fTTTzuN+/DDD5vOnTs71iWZv/zlL471uLg4I8nMnTvX0bZo0SJTvHjxPOutVauW+de//uVYj4iIMK+//rrT9unTpzvWe/ToYQYOHJjreMePHzeNGzc2kky1atXMgAEDzJIlS8yVK1ccfcLDw81LL72U4/5ff/218fDwMAkJCY62vXv3Gknmu+++M8YYM3HiROPl5WWSkpIcfdasWWMCAgLMxYsXncarXLmymTNnTp7nAIB744odgCKxf/9+fffdd+rbt68kydPTU48++miOty3r1Knj+O+wsDBJUlJSkiTpxx9/1IMPPujU/8EHH9SPP/6Y6xghISGSpPvuu8+p7eLFi0pJSZF09Tbx888/r1q1aqlEiRK655579NNPP+V6xU66etUuJibGUd+KFSucri5eLywsTHFxcdqzZ4+ee+45Xb58WQMGDFDHjh2VmZmppKQkHT9+XG3atMlx/x9//FHly5dX+fLlHW1Z9V47/4iICJUpU8axvn37dp07d05BQUG65557HMvhw4edbkcDuPPw8gSAIjF37lxlZGSobNmyjjZjjLy8vHT69GmVLFnS0X7tw/42m02SlJmZma3t2nGub8tpjLzG/dOf/qRVq1bpH//4h6pUqSIfHx/16dNHly5dynVO/fv314svvqi4uDjFxcWpYsWKat68+Q3OhBQZGanIyEiNGDFC33zzjZo3b64NGzaoYcOGee6X0zxzavfz83PanpmZqbCwMK1fvz7bviVKlLhhvQDcF8EOwG2XkZGhBQsWaMaMGWrfvr3Ttt69e+vDDz90et4tLzVr1tQ333yj/v37O9o2b96smjVr3lKNmzZt0sCBA9WzZ09JV5+5O3LkSJ77BAUFqUePHoqJiVFcXJyeeuqpmz5urVq1JF29Yujv76+KFStqzZo1atWqVY59ExISdPToUcdVu3379uns2bN5zr9BgwZKTEyUp6enKlaseNM1AnBfBDsAt90XX3yh06dPa/DgwQoMDHTa1qdPH82dOzffwe5Pf/qTHnnkEcfD/59//rmWLl2a5xus+VGlShUtXbpU3bp1k81m01//+lenq4S5GTJkiLp27aorV65owIABefZ99tlnFR4ertatW6tcuXI6ceKEJk+erDJlyqhJkyaSrr7VOmzYMAUHB6tTp05KTU3V//73P40aNUpt27ZVnTp19Pjjj2vWrFnKyMjQ8OHDFRUVlefVvrZt26pJkybq0aOHpk+frurVq+v48eP68ssv1aNHjxteKQTgvnjGDsBtN3fuXLVt2zZbqJOuXrHbtWuXduzYka+xevTooTfeeEOvvfaaateurTlz5igmJkYtW7a8pRpff/11lSxZUk2bNlW3bt3UoUMHNWjQ4Ib7tW3bVmFhYerQoYPCw8Nv2HfLli16+OGHVa1aNfXu3VvFixfXmjVrFBQUJEkaMGCAZs2apbffflu1a9dW165ddfDgQUlXbx9/9tlnKlmypFq0aKG2bdvq3nvv1ZIlS/I8rs1m05dffqkWLVpo0KBBqlatmvr27asjR444nj8EcGeyGZPDZwsAAFxy/vx5hYeH6/3331evXr2KuhwAdxluxQJAAcjMzFRiYqJmzJihwMBAPfTQQ0VdEoC7EMEOAApAQkKCKlWqpHLlymnevHny9OTXK4Dbj1uxAAAAFsHLEwAAABZBsAMAALAIgh0AAIBFEOwAAAAsgmAHAABgEQQ7AAAAiyDYAQAAWATBDgAAwCIIdgAAABbx/wABYBigYbDyZAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "異常検知の閾値 (95%点): 51153.6805\n"
     ]
    }
   ],
   "source": [
    "#Variance Estimation and Thresholding\n",
    "ssample = 100  \n",
    "patch_size = 30  \n",
    "num_features_per_patch = 32 * 15 * 15  \n",
    "M = num_features_per_patch  \n",
    "\n",
    "\n",
    "\n",
    "train_patches = []\n",
    "for i in range(ssample):\n",
    "    img = random.choice(train_images)  \n",
    "    x, y = np.random.randint(0, img.shape[1] - patch_size), np.random.randint(0, img.shape[0] - patch_size)\n",
    "    train_patches.append(img[y:y+patch_size, x:x+patch_size])\n",
    "\n",
    "train_patches = np.array(train_patches)  # (ssample, 30, 30)\n",
    "\n",
    "\n",
    "mean_pixel = np.mean(train_patches)\n",
    "std_pixel = np.std(train_patches)\n",
    "std_pixel = np.where(std_pixel == 0, 1e-8, std_pixel)  \n",
    "train_patches_normalized = (train_patches - mean_pixel) / std_pixel  # (ssample, 30, 30)\n",
    "\n",
    "\n",
    "\n",
    "train_patches_vectorized = train_patches_normalized.reshape(ssample, -1)  # (100, 900)\n",
    "\n",
    "\n",
    "mean_vector = np.mean(train_patches_vectorized, axis=0)  # (900,)\n",
    "\n",
    "\n",
    "centered_data = train_patches_vectorized - mean_vector  # (100, 900)\n",
    "\n",
    "\n",
    "cov_matrix = np.cov(centered_data, rowvar=False, ddof=1)  # (900, 900)\n",
    "\n",
    "\n",
    "num_patches_per_test = n + 1  \n",
    "num_features_per_patch = 30 * 30  \n",
    "total_features = num_patches_per_test * num_features_per_patch  # (n+1) * 900\n",
    "\n",
    "\n",
    "variances = np.var(train_patches_vectorized, axis=0, ddof=1)  # (900,)\n",
    "\n",
    "\n",
    "cov_matrix_diag = np.zeros((total_features, total_features))  # (n+1)*900 x (n+1)*900 のゼロ行列\n",
    "np.fill_diagonal(cov_matrix_diag, np.tile(variances, num_patches_per_test))  # (n+1)回分の分散を繰り返し適用\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "class SimpleAlexNet(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(SimpleAlexNet, self).__init__()\n",
    "        self.conv1 = nn.Conv2d(1, 32, kernel_size=5, stride=1, padding=2)\n",
    "        self.pool = nn.MaxPool2d(kernel_size=2, stride=2)\n",
    "        self.fc = nn.Linear(32 * 15 * 15, 10)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = F.relu(self.conv1(x))\n",
    "        x = self.pool(x)\n",
    "        x = x.view(x.size(0), -1)\n",
    "        x = self.fc(x)\n",
    "        return x\n",
    "\n",
    "\n",
    "model = SimpleAlexNet()\n",
    "model.eval()\n",
    "\n",
    "\n",
    "feature_list = []\n",
    "for i in range(ssample):\n",
    "    patch_tensor = torch.tensor(train_patches_normalized[i].reshape(1, 1, 30, 30), dtype=torch.float32)\n",
    "    with torch.no_grad():\n",
    "        feature_map = model.conv1(patch_tensor)\n",
    "        feature_map = model.pool(feature_map)\n",
    "        feature_vector = feature_map.view(-1).numpy()  # (7200,)\n",
    "    feature_list.append(feature_vector)\n",
    "\n",
    "feature_matrix = np.array(feature_list)  # (ssample, 7200)\n",
    "\n",
    "\n",
    "def f(X):\n",
    "    distances = []\n",
    "    for p in X[:, :M]:  \n",
    "        test_X = X[0, :M]  \n",
    "        distances.append(np.linalg.norm(test_X - p, ord=1))  \n",
    "    \n",
    "    distances.sort()\n",
    "    ip = distances[k]  \n",
    "    \n",
    "    \n",
    "    ip = max(ip, 1e-8)  \n",
    "\n",
    "    \n",
    "    a = (-1) * np.log(k) + M * np.log(ip)\n",
    "    a = np.clip(a, -100000, 100000)  \n",
    "    return a\n",
    "\n",
    "\n",
    "anomaly_scores = [f(feature_matrix[i:i+10]) for i in range(ssample - 10)]  \n",
    "\n",
    "\n",
    "finite_scores = [s for s in anomaly_scores if np.isfinite(s)]  \n",
    "plt.hist(finite_scores, bins=30, alpha=0.7, color='blue', edgecolor='black')\n",
    "plt.xlabel(\"Anomaly Score\")\n",
    "plt.ylabel(\"Frequency\")\n",
    "plt.title(\"Distribution of Anomaly Scores\")\n",
    "plt.show()\n",
    "\n",
    "\n",
    "a_th = np.percentile(finite_scores, 95)  \n",
    "\n",
    "print(f\"(95 percentile points): {a_th:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'num_abnormal_patches' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[1], line 33\u001b[0m\n\u001b[1;32m     28\u001b[0m         predicted_label \u001b[38;5;241m=\u001b[39m \u001b[38;5;241m0\u001b[39m\n\u001b[1;32m     30\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m predicted_label, a, ip\n\u001b[0;32m---> 33\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m i \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(\u001b[43mnum_abnormal_patches\u001b[49m):\n\u001b[1;32m     35\u001b[0m     feature_vector_set \u001b[38;5;241m=\u001b[39m lat_XX_mat[i, :\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m]\u001b[38;5;241m.\u001b[39mreshape(n\u001b[38;5;241m+\u001b[39m\u001b[38;5;241m1\u001b[39m, M)  \n\u001b[1;32m     38\u001b[0m     anomaly_score \u001b[38;5;241m=\u001b[39m f(feature_vector_set)\n",
      "\u001b[0;31mNameError\u001b[0m: name 'num_abnormal_patches' is not defined"
     ]
    }
   ],
   "source": [
    "\n",
    "y_list = []\n",
    "\n",
    "# lat_XX_mat （num_abnormal_patches, (n+1) * 7200 + 1）\n",
    "num_features_per_patch = 32 * 15 * 15  \n",
    "M = num_features_per_patch  \n",
    "\n",
    "\n",
    "\n",
    "def f(X):\n",
    "    distances = []\n",
    "    for p in X[1:, :M]:  \n",
    "        test_X = X[0, :M]  \n",
    "        distances.append(np.linalg.norm(test_X - p, ord=1))  \n",
    "    \n",
    "    distances.sort()\n",
    "    ip = distances[k]  \n",
    "\n",
    "    \n",
    "    ip = max(ip, 1e-8)  \n",
    "\n",
    "    \n",
    "    a = (-1) * np.log(k) + M * np.log(ip)\n",
    "    a = np.clip(a, -1000000, 1000000)  \n",
    "\n",
    "    if a > a_th :\n",
    "        predicted_label = 1    \n",
    "    else:\n",
    "        predicted_label = 0\n",
    "\n",
    "    return predicted_label, a, ip\n",
    "\n",
    "\n",
    "for i in range(num_abnormal_patches):\n",
    "    \n",
    "    feature_vector_set = lat_XX_mat[i, :-1].reshape(n+1, M)  \n",
    "\n",
    "    \n",
    "    anomaly_score = f(feature_vector_set)\n",
    "\n",
    "    y,_,_ = f(feature_vector_set)\n",
    "    y_list.append(y)\n",
    "\n",
    "\n",
    "print(f\"y_list size: {len(y_list)}\")\n",
    "print(f\"anomaly pecentile: {np.mean(y_list):.2%}\")  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'tqdm' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[2], line 7\u001b[0m\n\u001b[1;32m      4\u001b[0m p_list \u001b[38;5;241m=\u001b[39m []\n\u001b[1;32m      5\u001b[0m rejection \u001b[38;5;241m=\u001b[39m []\n\u001b[0;32m----> 7\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m i, (y, xx_list) \u001b[38;5;129;01min\u001b[39;00m \u001b[43mtqdm\u001b[49m(\u001b[38;5;28menumerate\u001b[39m(\u001b[38;5;28mzip\u001b[39m(y_list, XX_mat[:, :\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m]))):\n\u001b[1;32m      9\u001b[0m     xx_list \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39marray(xx_list)\n\u001b[1;32m     10\u001b[0m     eta \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39mzeros(\u001b[38;5;28mlen\u001b[39m(xx_list))\n",
      "\u001b[0;31mNameError\u001b[0m: name 'tqdm' is not defined"
     ]
    }
   ],
   "source": [
    "from cmath import e\n",
    "\n",
    "\n",
    "p_list = []\n",
    "rejection = []\n",
    "\n",
    "for i, (y, xx_list) in tqdm(enumerate(zip(y_list, XX_mat[:, :-1]))):\n",
    "   \n",
    "    xx_list = np.array(xx_list)\n",
    "    eta = np.zeros(len(xx_list))\n",
    "    distances = []\n",
    "    X = lat_XX_mat[:, :-1]\n",
    "\n",
    "    test_x = X[0]\n",
    "    for x in X:\n",
    "        distances.append(np.linalg.norm(test_x - x, ord=1))\n",
    "    newdistances = np.sort(distances)\n",
    "    indices = np.argsort(distances)\n",
    "    k_indices = indices[1:k+1]\n",
    "    y,_,_ = f(X)\n",
    "\n",
    "\n",
    "    import numpy as np\n",
    "\n",
    "\n",
    "    num_features_per_patch = 32 * 15 * 15  \n",
    "\n",
    "\n",
    "    knn_indices_list = []\n",
    "\n",
    "\n",
    "    def find_knn_indices(feature_vector_set):\n",
    "        \n",
    "\n",
    "    \n",
    "        distances = []\n",
    "        test_X = feature_vector_set[0, :num_features_per_patch]  \n",
    "\n",
    "\n",
    "        for i in range(1, len(feature_vector_set)):  \n",
    "            p = feature_vector_set[i, :num_features_per_patch]\n",
    "            dist = np.linalg.norm(test_X - p, ord=1)  \n",
    "            distances.append((dist, i))  \n",
    "\n",
    "\n",
    "        distances.sort()\n",
    "        knn_indices = [idx for _, idx in distances[:k]]  \n",
    "\n",
    "        return knn_indices\n",
    "\n",
    "\n",
    "    for i in range(num_abnormal_patches):\n",
    "        feature_vector_set = lat_XX_mat[i, :-1].reshape(n+1, num_features_per_patch)  # (n+1, 7200)\n",
    "        knn_indices = find_knn_indices(feature_vector_set)\n",
    "        knn_indices_list.append(knn_indices)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    num_train_samples = 100  \n",
    "    num_features = 900  \n",
    "\n",
    "    \n",
    "    train_patches = XX_mat[:, :-1]  \n",
    "    labels = XX_mat[:, -1]  \n",
    "    \n",
    "\n",
    "\n",
    "    num_features_per_patch = 30 * 30 \n",
    "    \n",
    "    \n",
    "\n",
    "\n",
    "    def compute_eta(X, knn_indices):\n",
    "        \"\"\"\n",
    "        X: (n+1, 900) \n",
    "        knn_indices: k \n",
    "        \n",
    "        \"\"\"\n",
    "\n",
    "        mean_test = np.mean(X[0, num_features_per_patch])  \n",
    "\n",
    "\n",
    "        mean_knn = np.mean([np.mean(X[(j-1)*(num_features_per_patch):(j)*num_features_per_patch]) for j in knn_indices]) \n",
    "\n",
    "\n",
    "        eta = (1 / num_features_per_patch) * np.ones(num_features_per_patch) - (1 / k) * (1 / num_features_per_patch) * np.ones(num_features_per_patch)\n",
    "\n",
    "        return eta\n",
    "    \n",
    "    eta = compute_eta(XX_mat, knn_indices)\n",
    "    P = eta @ eta.T\n",
    "    \n",
    "\n",
    "    \n",
    "\n",
    "\n",
    "    #Selection Event\n",
    "    def algorithm(a, b, z):\n",
    "        new_xx_list = a + b * z\n",
    "        renew_X = new_xx_list.reshape(30, 30) \n",
    "\n",
    "        \n",
    "\n",
    "\n",
    "        feature_list = []\n",
    "\n",
    "\n",
    "        num_samples = XX_mat.shape[0]  \n",
    "\n",
    "\n",
    "        model.eval()  \n",
    "\n",
    "    \n",
    "\n",
    "\n",
    "        patch_image = renew_X  \n",
    "\n",
    "\n",
    "        patch_tensor = torch.tensor(patch_image, dtype=torch.float32).unsqueeze(0)  # (1, 1, 30, 30)\n",
    "\n",
    "\n",
    "        with torch.no_grad():\n",
    "            feature_map = model.conv1(patch_tensor)  # (1, 32, 30, 30)\n",
    "            feature_vector = feature_map.view(-1).numpy()  \n",
    "\n",
    "        renew_XX_mat = np.vstack([feature_vector, XX_mat[:, :-1]]) \n",
    "        label,_,_ = f(renew_XX_mat)\n",
    "\n",
    "\n",
    "        feature_list.append(np.append(feature_vector, label))  \n",
    "\n",
    "\n",
    "        feature_matrix = np.array(feature_list)\n",
    "        \n",
    "\n",
    "\n",
    "        distances2 = []\n",
    "        new_test_X = feature_matrix[0, :-1]\n",
    "        new_X = feature_matrix[:, :-1]\n",
    "\n",
    "\n",
    "\n",
    "        for x in new_X:\n",
    "            distances2.append(np.linalg.norm(new_test_X - x,ord=1))\n",
    "        newdistances2 = np.sort(distances2)\n",
    "        indices2 = np.argsort(distances2)\n",
    "        k_indices2 = indices2[1:k+1]\n",
    "\n",
    "        \n",
    "        \n",
    "        _,a_x,_ = f(new_X)\n",
    "\n",
    "\n",
    "        inter_deep = [[-10, 10]]\n",
    "\n",
    "\n",
    "        X_data = XX_mat[:, :-1]  \n",
    "        num_features = X_data.shape[1]  \n",
    "\n",
    "        #DNN Polytope\n",
    "        W_conv = nn.Conv2d(32, num_features) \n",
    "        b_conv = np.random.randn(32)  \n",
    "\n",
    "        P = np.random.rand(7200, 32)  \n",
    "\n",
    "        W_fc = np.random.randn(10, 7200)  \n",
    "        b_fc = np.random.randn(10)  \n",
    "\n",
    "        # 3. A, b, c \n",
    "        A = W_conv.T @ P.T @ W_fc.T @ W_fc @ P @ W_conv  \n",
    "        b = W_conv.T @ P.T @ W_fc.T @ b_fc + W_conv.T @ b_conv  \n",
    "        c = b_fc.T @ b_fc  \n",
    "\n",
    "        inter_deep = sicore.polytope_to_interval(a, b, A=A, b=b, c=c)\n",
    "\n",
    "\n",
    "\n",
    "        #thresholding polytope\n",
    "        if a_x > a_th:\n",
    "            r = 1\n",
    "        else:\n",
    "            r = -1\n",
    "\n",
    "        b_i0 = np.zeros(len(new_xx_list))\n",
    "        for l in range(0, M):\n",
    "            n_ll = X[0][l] - X[indices2[k]][l]\n",
    "            if n_ll > 0:\n",
    "                t = 1\n",
    "            else:\n",
    "                t = -1\n",
    "            \n",
    "            b_i0[l] += 1 * t * (-r)\n",
    "            b_i0[l + M * indices[k]] -= 1 * t * (-r)\n",
    "        \n",
    "\n",
    "        c_i = math.pow(k * np.exp(a_th), 1/M) * r\n",
    "\n",
    "        inter1=sicore.polytope_to_interval(a, b, b=b_i0, c=c_i)\n",
    "\n",
    "\n",
    "\n",
    "        inter_near = [[-10, 10]]\n",
    "        inter_far = [[-10, 10]]\n",
    "        \n",
    "        knear = indices2[1:k]\n",
    "        kfar = indices2[k+1:]\n",
    "\n",
    "\n",
    "        new_near_lists = []\n",
    "        new_near_listk = []\n",
    "\n",
    "        for s in knear:\n",
    "            b_i1 = np.zeros(len(new_xx_list))\n",
    "\n",
    "            for l in range(0, M):\n",
    "                near_s = new_X[0][l] - new_X[s][l]\n",
    "                new_near_lists.append(near_s)\n",
    "                if near_s > 0:\n",
    "                    t = 1\n",
    "                else:\n",
    "                    t = -1\n",
    "                \n",
    "                b_i1[l] += 1 * t\n",
    "                b_i1[l + M * s] -= 1 * t\n",
    "\n",
    "                near_k = new_X[0][l] - new_X[indices2[k]][l]\n",
    "                new_near_listk.append(near_k)\n",
    "                if near_k > 0:\n",
    "                    tt = 1\n",
    "                else:\n",
    "                    tt = -1\n",
    "\n",
    "                b_i1[l] -= 1 * tt\n",
    "                b_i1[l + M * indices2[k]] += 1 * tt\n",
    "\n",
    "            inters = sicore.polytope_to_interval(a, b, b=b_i1)\n",
    "            inter_near = sicore.intersection(inter_near, inters)                \n",
    "\n",
    "        \n",
    "\n",
    "        new_far_listp = []\n",
    "        new_far_listk = []\n",
    "        for p in kfar:\n",
    "            b_i2 = np.zeros(len(new_xx_list))\n",
    "            for q in range(0, M):\n",
    "                far_q = new_X[0][q] - new_X[p][q]\n",
    "                new_far_listp.append(far_q)\n",
    "                if far_q > 0:\n",
    "                    r = 1\n",
    "                else:\n",
    "                    r = -1\n",
    "                \n",
    "                b_i2[q] -= 1 * r\n",
    "                b_i2[q + M * p] += 1 * r\n",
    "\n",
    "                far_k = new_X[0][q] - new_X[indices2[k]][q]\n",
    "                new_far_listk.append(far_k)\n",
    "                if far_k > 0:\n",
    "                    rr = 1\n",
    "                else:\n",
    "                    rr = -1\n",
    "                \n",
    "                b_i2[q] += 1 * rr\n",
    "                b_i2[q + M * indices2[k]] -= 1 * rr\n",
    "            \n",
    "            interp = sicore.polytope_to_interval(a, b, b=b_i2)\n",
    "            inter_far = sicore.intersection(inter_far, interp)\n",
    "\n",
    "        \n",
    "        inter_distance = sicore.intersection(inter_near, inter_far)\n",
    "        inter = sicore.intersection(inter1, inter_distance)\n",
    "        inter = sicore.intersection(inter, inter_deep)\n",
    "        \n",
    "        assert inter[0][0] < z < inter[0][1]\n",
    "        return (a_x, indices2[k]), inter\n",
    "    def modelselector(yy):\n",
    "        if y == 1:\n",
    "            if yy[0] > a_th and indices[k] == yy[1]:\n",
    "                return True\n",
    "            else:\n",
    "                return False\n",
    "        else :\n",
    "            if yy[0] < a_th and indices[k] ==yy[1]:\n",
    "                return True\n",
    "            else:\n",
    "                return False\n",
    "    \n",
    "    SI = sicore.SelectiveInferenceChi(xx_list[0:num_features], cov_matrix_diag, P=P)\n",
    "\n",
    "    try:\n",
    "        result = SI.inference(algorithm=algorithm, model_selector=modelselector, alternative=\"less\", step=1e-10, over_conditioning=True, max_iter=100)\n",
    "        p_list.append(result.p_value)\n",
    "        rejection.append(result.reject_or_not)\n",
    "    except Exception as e:\n",
    "        print(i)\n",
    "        print(e)\n",
    "    \n",
    "pwr = sicore.power(p_list, alpha=0.05)\n",
    "print(pwr)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "#type I error experiments(You should use only normal data)\n",
    "typeIerror = sicore.typeIerror(p_list, alpha=0.05)\n",
    "print(typeIerror)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'sicore' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[3], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m pwr \u001b[38;5;241m=\u001b[39m \u001b[43msicore\u001b[49m\u001b[38;5;241m.\u001b[39mpower(p_list, alpha\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m0.05\u001b[39m)\n\u001b[1;32m      2\u001b[0m \u001b[38;5;28mprint\u001b[39m(pwr)\n",
      "\u001b[0;31mNameError\u001b[0m: name 'sicore' is not defined"
     ]
    }
   ],
   "source": [
    "#Power experiments\n",
    "pwr = sicore.power(p_list, alpha=0.05)\n",
    "print(pwr)"
   ]
  }
 ],
 "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.10.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
