{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "import numpy as np"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# test function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def fnOgHimmelblau(x,y):\n",
    "    return np.square(x*x + y - 11) + np.square(x + y*y - 7)\n",
    "def gradOgHimmelblau(x,y):\n",
    "    return 4*x*(x*x + y - 11) + 2*(x + y*y - 7), 2*(x*x + y - 11) + 4*y*(x + y*y - 7)\n",
    "\n",
    "def fnTransformforHimmelblau(x,y,const=0.5,ref_x = 0.5):\n",
    "    return x*(np.exp(const*(x - ref_x) )+1),y*(np.exp(const*(x - ref_x) )+1)\n",
    "def fnTransformedHimmelblau(x,y,const=0.5,ref_x = 1.5,offset= 1.5):\n",
    "    xNew,yNew = fnTransformforHimmelblau(x - offset,y,const,ref_x)\n",
    "    return fnOgHimmelblau(xNew,yNew)\n",
    "\n",
    "def jacobTransformation(x,y,const=0.5,ref_x = 1.5,offset= 1.5):\n",
    "    termExponential = np.exp( const * (x - offset - ref_x) )\n",
    "    \n",
    "    J = np.array( [ [ termExponential + 1 + (x - offset)*termExponential*const, 0  ],\n",
    "                   [ y*termExponential*const, 1 + termExponential  ] ] )\n",
    "    \n",
    "    return J\n",
    "\n",
    "def gradTransformedHimmelblau(x,y,const=0.5,ref_x = 1.5,offset= 2.0):\n",
    "    xNew,yNew = fnTransformforHimmelblau(x - offset,y,const,ref_x)\n",
    "    g = gradOgHimmelblau(xNew,yNew)\n",
    "    J = jacobTransformation(x,y,const,ref_x,offset)\n",
    "    \n",
    "    return np.matmul(g,J)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def fnCutHimmelblau(x,y,const=0.5,ref_x = 1.5,offset= 1.5,\n",
    "                    cut_endpoint = -0.5,cut_length = 5,\n",
    "                   center_y = -3.23360726,slope = 10,height = 4.2,bound_y = 2.0):\n",
    "    \n",
    "    flag_left = x < cut_endpoint - cut_length\n",
    "    flag_right = x > cut_endpoint\n",
    "    flag_in = (1 - np.maximum(flag_left,flag_right)) * ( np.abs(y - center_y) < bound_y )\n",
    "    \n",
    "    ogValue = fnTransformedHimmelblau(x, y, const, ref_x, offset)\n",
    "    return (1 - flag_in)* ogValue + flag_in * np.maximum(ogValue, height + slope*np.power( np.abs(y - center_y),1.1 )   )\n",
    "\n",
    "def gradCutHimmelblau(x,y,const=0.5,ref_x = 1.5,offset= 1.5,\n",
    "                    cut_endpoint = -0.5,cut_length = 5,\n",
    "                   center_y = -3.23360726,slope = 10,height = 4.2,bound_y = 2.0):\n",
    "    \n",
    "    flag_left = x < cut_endpoint - cut_length\n",
    "    flag_right = x > cut_endpoint\n",
    "    flag_in = (1 - np.maximum(flag_left,flag_right)) * ( np.abs(y - center_y) < bound_y )\n",
    "    \n",
    "    ogValue = fnTransformedHimmelblau(x, y, const, ref_x, offset)\n",
    "    flag = flag_in * (ogValue < height + slope*np.power( np.abs(y - center_y),1.1 ) )\n",
    "    \n",
    "    ogGrad = gradTransformedHimmelblau(x,y,const,ref_x,offset)\n",
    "    diff_y = y - center_y\n",
    "    \n",
    "    return flag * np.array([0.0,slope]) * 1.1*np.power( np.abs(diff_y),0.1 )*np.sign(diff_y) + (1 - flag) * ogGrad "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# utilities"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def generateUniformDistOnSphere(thisD):\n",
    "    if thisD == 1:\n",
    "        return np.random.choice([-1,1])\n",
    "    \n",
    "    \n",
    "    v = np.random.uniform( size = thisD - 1 )*np.pi\n",
    "    v[-1]*= 2\n",
    "    \n",
    "    listCosine = np.cos(v)\n",
    "    listSine = np.sin(v)\n",
    "    \n",
    "    output = np.zeros(thisD)\n",
    "    for idx in range(thisD - 1):\n",
    "        output[idx] = np.prod( listSine[:idx] )*listCosine[idx]\n",
    "    \n",
    "    output[-1] = np.prod(listSine)\n",
    "    \n",
    "    return output\n",
    "        "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def generateHeavyTailNoiseConeVersion(D,listDirections,listHeavyTailIndex,listMagnitudeNoise):\n",
    "    \n",
    "    output = np.zeros(D)\n",
    "    \n",
    "    for thisDirectionList,thisAlpha,thisMagnitudeNoise in zip(listDirections,listHeavyTailIndex,listMagnitudeNoise):\n",
    "        noiseNorm = thisMagnitudeNoise * np.random.pareto(thisAlpha)\n",
    "        tempProjection = generateUniformDistOnSphere( thisDirectionList.size )\n",
    "        tempNoise = noiseNorm * tempProjection\n",
    "        \n",
    "        output[thisDirectionList] = tempNoise\n",
    "    \n",
    "    return output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def stepSgdConeVersion(xNow,lr,D,listDirections,listHeavyTailIndex,listMagnitudeNoise,\n",
    "                       clipping_threshold = None,bound = 5.0,g_scale = 1.0,\n",
    "                       const=0.5,ref_x = 1.5,offset= 1.5,\n",
    "                       cut_endpoint = -0.5,cut_length = 5,\n",
    "                       center_y = -3.23360726,slope = 10,height = 4.2,bound_y = 2.0):\n",
    "    \n",
    "    g = gradCutHimmelblau(xNow[0],xNow[1],const,ref_x,offset,\n",
    "                          cut_endpoint,cut_length,\n",
    "                          center_y ,slope,height,bound_y)\n",
    "    \n",
    "    noise = generateHeavyTailNoiseConeVersion(D,listDirections,listHeavyTailIndex,listMagnitudeNoise)\n",
    "    gNoisy = (g*g_scale + noise)*lr\n",
    "    \n",
    "    # gradient clipping\n",
    "    if clipping_threshold is not None:\n",
    "        normGradNoisy = np.linalg.norm(gNoisy)\n",
    "        if normGradNoisy > clipping_threshold:\n",
    "            gNoisy *= clipping_threshold/normGradNoisy\n",
    "    \n",
    "    # apply noisy gradient\n",
    "    xOut = xNow - gNoisy\n",
    "    \n",
    "    # reflection at boundary\n",
    "    # xOut = np.maximum( -bound, np.minimum(bound, xOut ) )\n",
    "    normOut = np.linalg.norm(xOut)\n",
    "    if normOut > bound:\n",
    "        xOut *= bound/normOut\n",
    "    \n",
    "    return xOut"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def loopSgdConeVersion(xInit,lr,numIter,D,\n",
    "                       listDirections,listHeavyTailIndex,listMagnitudeNoise,\n",
    "                       clipping_threshold = None,bound = 5.0,g_scale = 1.0,\n",
    "                       const=0.5,ref_x = 1.5,offset= 1.5,\n",
    "                       cut_endpoint = -0.5,cut_length = 5,\n",
    "                       center_y = -3.23360726,slope = 10,height = 4.2,bound_y = 2.0):\n",
    "    \n",
    "    listDistance = np.zeros(numIter)\n",
    "    tableIterates = np.zeros([numIter,D])\n",
    "    xNow = np.copy(xInit)\n",
    "    \n",
    "    for idx in range(numIter):\n",
    "        if idx % 1000000 == 0:\n",
    "            print(idx)\n",
    "        xNow = stepSgdConeVersion(xNow,lr,D,listDirections,listHeavyTailIndex,listMagnitudeNoise,\n",
    "                                  clipping_threshold,bound,g_scale,\n",
    "                                  const,ref_x,offset,\n",
    "                                  cut_endpoint,cut_length,\n",
    "                                  center_y ,slope,height,bound_y)\n",
    "        \n",
    "        listDistance[idx] = np.linalg.norm( xNow )\n",
    "        tableIterates[idx] = xNow\n",
    "    \n",
    "    return listDistance,tableIterates"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Experiment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "D = 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "xInit = np.zeros(D)\n",
    "xInit[0] = 2.9\n",
    "xInit[1] = 1.0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "numIter = 30000000\n",
    "b = 2.15\n",
    "lr = 5e-4\n",
    "bound = 4.2\n",
    "g_scale = 0.1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "_,iteratesClipping = loopSgdConeVersion(xInit,lr,numIter,D,\n",
    "                                                       listDirections,listHeavyTailIndex,listMagnitudeNoise,\n",
    "                                                       clipping_threshold = b,bound = bound,g_scale = g_scale,\n",
    "                                                       const=0.4,ref_x = 1.5,offset = 1.4, slope = 12, center_y=-2.9)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "_,iteratesNoClipping = loopSgdConeVersion(xInit,lr,numIter,D,\n",
    "                                                       listDirections,listHeavyTailIndex,listMagnitudeNoise,\n",
    "                                                       clipping_threshold = None,bound = bound,g_scale = g_scale,\n",
    "                                                       const=0.4,ref_x = 1.5,offset = 1.4, slope = 12, center_y=-2.9)"
   ]
  }
 ],
 "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
}
