{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pickle"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# test function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def f1(x):\n",
    "    return x+1.6\n",
    "\n",
    "def g1(x):\n",
    "    return 1\n",
    "\n",
    "def f2(x):\n",
    "    return (x+1.3)**2\n",
    "\n",
    "def g2(x):\n",
    "    return 2*(x+1.3)\n",
    "\n",
    "def f3(x):\n",
    "    return (x-0.2)**2\n",
    "\n",
    "def g3(x):\n",
    "    return 2*(x-0.2)\n",
    "\n",
    "def f4(x):\n",
    "    return (x-0.7)**2\n",
    "\n",
    "def g4(x):\n",
    "    return 2*(x-0.7)\n",
    "\n",
    "def f5(x):\n",
    "    return x-1.6\n",
    "\n",
    "def g5(x):\n",
    "    return 1\n",
    "\n",
    "def f6(x):\n",
    "    return ( 0.05 * np.abs(x - 1.65) )**0.6\n",
    "\n",
    "def g6(x):\n",
    "    return -(0.05**0.6)*( 0.6*(1.65 - x)**(-0.4) )\n",
    "\n",
    "def f7(x):\n",
    "    return 1 + ( 0.01 + 4*((x-0.5)**2)  )**(-1)\n",
    "\n",
    "def g7(x):\n",
    "    return -( 8 * (x-.5) )/( ( 0.01 + 4*((x-0.5)**2)  )**2  )\n",
    "\n",
    "def f8(x):\n",
    "    return 1 + ( 0.1 + 4*((x+1.5)**2)  )**(-1)\n",
    "\n",
    "def g8(x):\n",
    "    return -( 8 * (x+1.5) )/( ( 0.1 + 4*((x+1.5)**2)  )**2 ) \n",
    "\n",
    "\n",
    "def f9(x):\n",
    "    return 1 - 0.25 * np.exp( -5 * (x + 0.8)*(x + 0.8)  )\n",
    "\n",
    "def g9(x):\n",
    "    return 2.5*np.exp( -5 * (x + 0.8)*(x + 0.8)  )*(x+0.8)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "listF = [f1,f2,f3,f4,f5,f6,f7,f8,f9]\n",
    "\n",
    "def f(x):\n",
    "    out = 1\n",
    "    for f_each in listF:\n",
    "        out = out * f_each(x)\n",
    "    return out\n",
    "\n",
    "# check gradient\n",
    "listG = [g1,g2,g3,g4,g5,g6,g7,g8,g9]\n",
    "\n",
    "def g(x):\n",
    "    \n",
    "    out_final = 0\n",
    "    \n",
    "    for idx_g,g_this in enumerate(listG):\n",
    "        out_this = g_this(x)\n",
    "        for idx_f,f_this in enumerate(listF):\n",
    "            if idx_f != idx_g:\n",
    "                out_this = out_this * f_this(x)\n",
    "        \n",
    "        out_final += out_this\n",
    "    \n",
    "    return out_final"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Utilites: SGD updates"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def SGDstep(x,lr,noise_type = None,noise_magnitude = 1,noise_shape = None,\n",
    "            gradient_clipping_threshold = None,reflection_boundary = None):\n",
    "    assert noise_type in [None,'normal','pareto']\n",
    "    if noise_type == 'pareto':\n",
    "        assert noise_shape is not None\n",
    "    \n",
    "    this_gradient = g(x)\n",
    "    \n",
    "    if noise_type is None:\n",
    "        this_noise = 0\n",
    "    elif noise_type is 'normal':\n",
    "        this_noise = noise_magnitude * np.random.normal(size=x.shape)\n",
    "    elif noise_type is 'pareto':\n",
    "        this_noise = noise_magnitude * np.random.pareto(noise_shape, size = x.shape)\n",
    "        this_sign = np.random.choice([-1,1],size = this_noise.shape)\n",
    "        this_noise = this_noise * this_sign\n",
    "        \n",
    "    this_step = lr*(-this_gradient + this_noise)\n",
    "    if gradient_clipping_threshold is not None:\n",
    "        this_step = np.minimum( np.maximum( this_step, - gradient_clipping_threshold ), gradient_clipping_threshold)\n",
    "    \n",
    "    x_out = x + this_step\n",
    "    if reflection_boundary is not None:\n",
    "        x_out = np.minimum( np.maximum(x_out,-reflection_boundary),reflection_boundary  )\n",
    "    \n",
    "    return x_out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def SGDtrain(x_initial,n_step,\n",
    "             lr,n_run = 1,noise_type = None,noise_magnitude = 1,noise_shape = None,\n",
    "             gradient_clipping_threshold = None, reflection_boundary = None ):\n",
    "    \n",
    "    x_now = np.ones(n_run) * x_initial\n",
    "    \n",
    "    record = np.zeros( (n_step,n_run) )\n",
    "    \n",
    "    #record = []\n",
    "    \n",
    "    for j in range(n_step):\n",
    "        if j % 500000 == 0:\n",
    "            print(j)\n",
    "        x_now = SGDstep(x_now,lr,noise_type = noise_type, \n",
    "                        noise_magnitude = noise_magnitude,noise_shape = noise_shape,\n",
    "                        gradient_clipping_threshold = gradient_clipping_threshold,\n",
    "                        reflection_boundary = reflection_boundary)\n",
    "        \n",
    "        #record.append(x_now)\n",
    "        record[j,:] = x_now\n",
    "    return record"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def FirstExitTime(x_initial,exit_left,exit_right,lr,\n",
    "                  noise_type = None,noise_magnitude = 1,noise_shape = None,\n",
    "                  gradient_clipping_threshold = None,reflection_boundary = None,stop_iteration = None):\n",
    "    \n",
    "    assert exit_left < exit_right\n",
    "    \n",
    "    nTime = 0\n",
    "    x_now = np.array(x_initial)\n",
    "    while x_now > exit_left and x_now < exit_right:\n",
    "        x_now = SGDstep(x_now,lr,noise_type = noise_type, \n",
    "                        noise_magnitude = noise_magnitude,noise_shape = noise_shape,\n",
    "                        gradient_clipping_threshold = gradient_clipping_threshold,\n",
    "                        reflection_boundary = reflection_boundary)\n",
    "        nTime += 1\n",
    "        \n",
    "        if stop_iteration is not None and nTime > stop_iteration:\n",
    "            break\n",
    "    \n",
    "    return nTime"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Experiment 1: First Exit Time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "noise_type = 'pareto'\n",
    "noise_shape = 1.2\n",
    "noise_magnitude = 0.1\n",
    "\n",
    "#gradient_clipping_threshold = 0.5\n",
    "reflection_boundary = 1.6\n",
    "\n",
    "exit_left = -1.3\n",
    "exit_right = 0.2\n",
    "\n",
    "nsample = 20\n",
    "x_initial = -0.7"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "list_gradient_clipping_threshold = [None, 0.5, 0.28]\n",
    "list_lr = [ [0.1, 0.03, 0.01, 0.003, 0.001, 0.0003, 0.0001],\n",
    "          [0.1, 0.03, 0.01, 0.003, 0.001, 0.0003, 0.0001],\n",
    "          [0.1, 0.05, 0.03, 0.02, 0.01, 0.005, 0.003, 0.001]]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "list_records = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "for idx in range(3):\n",
    "    gradient_clipping_threshold = list_gradient_clipping_threshold[idx]\n",
    "    for lr in list_lr[idx]:\n",
    "        record = []\n",
    "        while len(record) < nsample:\n",
    "            this_exit_time = FirstExitTime(x_initial,exit_left,exit_right,lr,\n",
    "                                           noise_type = noise_type,noise_magnitude = noise_magnitude,noise_shape = noise_shape,\n",
    "                                           gradient_clipping_threshold = gradient_clipping_threshold,\n",
    "                                           reflection_boundary = reflection_boundary)\n",
    "            record.append(this_exit_time)\n",
    "        list_records.append(record)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Experiment 2: Path of SGD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "n_run = 10\n",
    "lr = 1e-3\n",
    "n_step = 10000000\n",
    "x_initial = 0.3\n",
    "reflection_boundary = 1.6"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "list_noise_type = ['pareto','normal']\n",
    "noise_shape = 1.2\n",
    "list_noise_magnitude = [0.1,1]\n",
    "list_gradient_clipping_threshold = [None, 0.5]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "list_path = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "for idx in range(2):\n",
    "    noise_type = list_noise_type[idx]\n",
    "    noise_magnitude = list_noise_magnitude[idx]\n",
    "    for gradient_clipping_threshold in list_gradient_clipping_threshold:\n",
    "        record = SGDtrain(x_initial,n_step,lr,\n",
    "                           n_run = n_run, noise_type = noise_type, \n",
    "                           noise_magnitude = noise_magnitude, noise_shape = noise_shape,\n",
    "                           gradient_clipping_threshold = gradient_clipping_threshold, \n",
    "                           reflection_boundary = reflection_boundary )\n",
    "        list_path.append( record )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "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.5.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
