{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import util"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "util.normalize_datasets(1000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "def read_libsvm_data(filename, num_features):\n",
    "    data = []\n",
    "    with open(filename, 'r') as f:\n",
    "        for line in f:\n",
    "            line = line.strip().split()\n",
    "            label = line[0]\n",
    "            features = [0.0 for _ in range(num_features+1)]\n",
    "            for i in range(1, len(line)):\n",
    "                line[i] = line[i].split(':')\n",
    "                features[int(line[i][0])] = float(line[i][1])\n",
    "            data.append(features)\n",
    "    return np.array(data)\n",
    "\n",
    "mushroom = read_libsvm_data(\"datasets/mushrooms.txt\",112)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "from util import get_diameter_pointdiff\n",
    "from tqdm import tqdm \n",
    "from numba import jit\n",
    "import time\n",
    "def normalize_datasets_approx(num_rays, dataset,name):\n",
    "    print(\"starting normalization\")\n",
    "    diam = get_diameter_pointdiff(dataset,num_rays)\n",
    "    dataset = (dataset/diam, name)\n",
    "    np.save(\"normalized_datasets/\"+dataset[1]+\"_normalized\", dataset[0])\n",
    "        \n",
    "def normalize_dataset_exact(dataset, name):\n",
    "    diam = util.find_diameter(dataset)\n",
    "    dataset = (dataset/diam, name)\n",
    "    np.save(\"normalized_datasets/\"+dataset[1]+\"_normalized\", dataset[0])\n",
    "    \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sklearn\n",
    "import numpy as np\n",
    "test_data = sklearn.datasets.make_blobs(n_samples=5000, n_features=6, centers=100, cluster_std=1.0, center_box=(-10.0, 10.0), shuffle=True, random_state=None)[0]\n",
    "normalized_test_data = normalize_dataset_exact( test_data, \"test_data\")\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "time for exact normalization:  0.0\n"
     ]
    }
   ],
   "source": [
    "start_time = time.time()\n",
    "# normalize_dataset_exact(mushroom, \"mushroom\")\n",
    "print(\"time for exact normalization: \", time.time() - start_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "skin_nonskin = read_libsvm_data(\"datasets/skin_nonskin.txt\",3)\n",
    "# normalize_datasets_approx(20000, skin_nonskin, \"skin_nonskin\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "starting normalization\n",
      "0\n",
      "updated max dist from  0.0  to  8727.494944140615\n",
      "updated max dist from  8727.494944140615  to  8794.970778803077\n",
      "updated max dist from  8794.970778803077  to  8833.16223104727\n",
      "updated max dist from  8833.16223104727  to  8833.534400227352\n",
      "updated max dist from  8833.534400227352  to  8835.561612031235\n",
      "updated max dist from  8835.561612031235  to  8851.720058836023\n",
      "updated max dist from  8851.720058836023  to  8853.366308924533\n",
      "2000\n",
      "4000\n",
      "6000\n",
      "8000\n",
      "10000\n",
      "12000\n",
      "14000\n",
      "16000\n",
      "18000\n"
     ]
    }
   ],
   "source": [
    "import sklearn.datasets as datasets\n",
    "import numpy as np\n",
    "# data = util.get_covtype()\n",
    "\n",
    "cov_type = datasets.fetch_covtype(download_if_missing=True, return_X_y=True, data_home=\"datasets/\")\n",
    "normalize_datasets_approx(20000, cov_type[0], \"covtype\")\n",
    "# normalize_datasets_approx(20000, cov_type.data, \"covtype\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "c:\\Users\\au616584\\Anaconda3\\envs\\subspace\\lib\\site-packages\\tqdm\\auto.py:22: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n"
     ]
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import torch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 107,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([[0.3589]], requires_grad=True) tensor([[0.0722]], requires_grad=True) tensor([[0.4994]], requires_grad=True)\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def find_best_fit(x,y,k):\n",
    "    c = torch.ones(1,1)#torch.from_numpy(np.array(y))[0]\n",
    "    k_z = torch.ones(1,1)\n",
    "    k_z -= 0.5\n",
    "    n_z = torch.ones(1,1) \n",
    "    n_z -= 0.3\n",
    "    c.requires_grad = True\n",
    "    k_z.requires_grad = True\n",
    "    n_z.requires_grad = True\n",
    "    # sgd = torch.optim.AdamW([c, n_z], lr=0.001)\n",
    "    # x_ = torch.from_numpy(np.array(x))\n",
    "    # y_ = torch.from_numpy(np.array(y))\n",
    "    # for epoch in range(10000):\n",
    "    #     loss = torch.sum((y_ - c*torch.pow(k,k_z)/torch.pow(x_,n_z))**2)\n",
    "    #     loss.backward()\n",
    "    #     sgd.step()\n",
    "    #     sgd.zero_grad()\n",
    "    # print(c,k_z,n_z)\n",
    "    sgd2 = torch.optim.AdamW([c,k_z, n_z], lr=0.001)\n",
    "    for epoch in range(10000):\n",
    "        x_ = torch.from_numpy(np.array(x))\n",
    "        y_ = torch.from_numpy(np.array(y))\n",
    "        loss = torch.sum((y_ - c*torch.pow(k,k_z)*torch.pow(x_,-n_z))**2)\n",
    "        loss.backward()\n",
    "        sgd2.step()\n",
    "        sgd2.zero_grad()\n",
    "    print(c,k_z,n_z)\n",
    "    c.requires_grad = False\n",
    "    k_z.requires_grad = False\n",
    "    n_z.requires_grad = False\n",
    "    # k_z.detach()\n",
    "    # n_z.detach()\n",
    "    return c,k_z,n_z\n",
    "        \n",
    "\n",
    "\n",
    "x = [i for i in range(100,1000)]\n",
    "k=20\n",
    "y = [0.1*(k**0.5)/(n**0.5) for n in x]\n",
    "\n",
    "c, k_z, n_z = find_best_fit(x,y,k)\n",
    "plt.plot(x, [c*(k**k_z)/(n**n_z) for n in x])\n",
    "plt.plot(x,y)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "normalize_dataset_exact(mushroom, \"mushroom\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "subspace",
   "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.8"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "3f94b3ea8b08becd1766c9f3a63569b610389292faace546d6478dffb17875b9"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
