{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matlab.engine"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.append('..')\n",
    "import torch\n",
    "import numpy as np\n",
    "\n",
    "\n",
    "from hyperbox import Hyperbox\n",
    "from interval_analysis import HBoxIA\n",
    "from relu_nets import ReLUNet\n",
    "from lipMIP import LipProblem\n",
    "from other_methods import CLEVER, FastLip, LipLP, LipSDP, NaiveUB, RandomLB, SeqLip\n",
    "from neural_nets import train\n",
    "from neural_nets import data_loaders as dl\n",
    "from experiment import Experiment, InstanceGroup, Result\n",
    "from utilities import Factory"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Basic test file to demonstrate that everything works \n",
    "# 1) Build dataset and display it \n",
    "# 2) Build neural net and train it \n",
    "# 3) Run single evaluations for a particular hyperbox and multiple types of results\n",
    "# 4) Run some simple experimental series\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 1) Build a dataset and display it \n",
    "dataset_params = dl.RandomKParameters(1024, 30, radius=0.02, dimension=4)\n",
    "dataset = dl.RandomBinaryDataset(dataset_params, random_seed=420)\n",
    "train_data, val_data = dataset.split_train_val(1.0)\n",
    "#dataset.plot_2d()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 2) Build neural net and train it \n",
    "test_net = ReLUNet(layer_sizes=[4, 16, 16, 2])\n",
    "# Build training loss function --- \n",
    "xentropy_term = train.XEntropyReg()\n",
    "l2_weight_term = train.LpWeightReg(scalar=1e-2, lp='l2')\n",
    "loss = train.LossFunctional(regularizers=[xentropy_term, l2_weight_term])\n",
    "train_params = train.TrainParameters(train_data, train_data, 1000, test_after_epoch=100)\n",
    "train.training_loop(test_net, train_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 3) Run single evaluations for a particular hyperbox and multiple types of results\n",
    "hbox1 = Hyperbox.build_linf_ball(np.array([0.5 for _ in range(test_net.layer_sizes[0])]), 0.2)\n",
    "c_vec = torch.Tensor([1.0, -1.0])\n",
    "local_args = {'network': test_net, 'domain': hbox1, 'c_vector': c_vec, 'primal_norm': 'linf'}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 3.1) lipMIP\n",
    "lip_prob = LipProblem(test_net, hbox1, c_vec, verbose=True, num_threads=4)\n",
    "lip_prob.compute_max_lipschitz()\n",
    "lip_prob.result"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 3.2) CLEVER\n",
    "clever = CLEVER(**local_args)\n",
    "clever.compute()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 3.3) FastLip\n",
    "fastlip = FastLip(**local_args)\n",
    "fastlip.compute()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 3.4) LipLP\n",
    "liplp = LipLP(**local_args)\n",
    "liplp.compute()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 3.5) LipSDP\n",
    "lipsdp = LipSDP(test_net, c_vec)\n",
    "lipsdp.compute()\n",
    "print(lipsdp.l1_value())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 3.6) Naive UB\n",
    "naiveub = NaiveUB(test_net, c_vec, 'linf')\n",
    "print(\"L1 norms of matrices:\", naiveub.compute())\n",
    "naiveub2 = NaiveUB(test_net, c_vec, 'l2')\n",
    "print(\"L2 norms of matrices, scaled by sqrt(2)\", naiveub2.compute() * 2**0.5)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 3.7) Random LB\n",
    "randomlb = RandomLB(**local_args)\n",
    "randomlb.compute()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 3.8) SeqLip\n",
    "seqlip = SeqLip(**local_args)\n",
    "seqlip.compute() * 4 **0.5 \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 4) Run some simple experimental series\n",
    "exp_1 = Experiment([RandomLB, FastLip, LipProblem], network=test_net, c_vector=c_vec, \n",
    "                   primal_norm='linf', verbose=False, num_threads=4)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 4.1) Evaluate over the unit hypercube\n",
    "hypercube_result = exp_1.do_unit_hypercube_eval()\n",
    "print(hypercube_result.values())\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 4.2) Evaluate over random Linf balls of radius 0.2\n",
    "ball_factory = Factory(Hyperbox.build_linf_ball, radius=0.2, global_lo=np.array([0,0]), global_hi=np.array([1.0, 1.0]))\n",
    "center_domain = Hyperbox.build_linf_ball(x=np.array([0.5, 0.5]), radius=0.5)\n",
    "random_results = exp_1.do_random_evals(10, center_domain, ball_factory)\n",
    "random_results.average_stdevs('value')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 4.3) Evaluate over random data points \n",
    "data_results = exp_1.do_data_evals(train_data[0][0], ball_factory, num_random=10)\n",
    "data_results.average_stdevs('value')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "torch.cuda.is_available()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<torch.utils.data.dataloader.DataLoader at 0x7fa7bd6d36d8>"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dl.load_mnist_data('train',  digits=[1,7], use_gpu=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.6.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
