{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "77592115",
   "metadata": {},
   "source": [
    "# Example 4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "438b0c06",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "checkSOSMonotone (generic function with 1 method)"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "using DynamicPolynomials\n",
    "include(\"../scripts/SOSGameClass.jl\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "ced5d85f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Payoff function for P1: 3 - 8*y - 19*x + 45*x*y + 49*x^2 - 89*x^2*y - 59*x^3 + 74*x^3*y + 25*x^4 - 16*x^4*y\n"
     ]
    }
   ],
   "source": [
    "@polyvar x\n",
    "@polyvar y\n",
    "\n",
    "# payoff for P1\n",
    "u = 5*x^4*y - 1*x^4*(1-y) - 5*x^3*(1-x)*y + 2*x^3*(1-x)*(1-y) + \n",
    "    3*x^3*(1-x)*y - 2*x^3*(1-x)*(1-y) + 4*x^2*(1-x)^2*y + 1*x^2*(1-x)^2*(1-y) + \n",
    "    (-3)*x^3*(1-x)*y - 2*x^3*(1-x)*(1-y) - 1*x^2*(1-x)^2*y + 2*x^2*(1-x)^2*(1-y) + \n",
    "    1*x^2*(1-x)^2*y - 3*x^2*(1-x)^2*(1-y) + 2*x*(1-x)^3*y - 2*x*(1-x)^3*(1-y) + \n",
    "    (-2)*x^3*(1-x)*y - 4*x^3*(1-x)*(1-y) + 4*x^2*(1-x)^2*y + 1*x^2*(1-x)^2*(1-y) + \n",
    "    2*x^2*(1-x)^2*y + 5*x^2*(1-x)^2*(1-y) - 5*x*(1-x)^3*y - 1*x*(1-x)^3*(1-y) + \n",
    "    (-2)*x^2*(1-x)^2*y + 4*x^2*(1-x)^2*(1-y) + 4*x*(1-x)^3*y + (-5)*x*(1-x)^3*(1-y) + \n",
    "    5*x*(1-x)^3*y + 1*x*(1-x)^3*(1-y) - 5*(1-x)^4*y + 3*(1-x)^4*(1-y)\n",
    "\n",
    "println(\"Payoff function for P1: \", u)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d9473a32",
   "metadata": {},
   "source": [
    "## Check the SOS-monotonicity of the original EFG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "5fd44e94",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Basic semialgebraic Set defined by no equality\n",
       "6 inequalities\n",
       " x ≥ 0\n",
       " y ≥ 0\n",
       " 1 - x ≥ 0\n",
       " 1 - y ≥ 0\n",
       " 1 - x^2 ≥ 0\n",
       " 1 - y^2 ≥ 0\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# construct the domain \n",
    "gs = [x; y; 1 - x; 1 - y; 1 - x^2; 1 - y^2]     # x >= 0; y >= 0; 1 - x >= 0; 1 - y >= 0; 1 - x^2 >= 0; 1 - y^2 >= 0\n",
    "Sx = basic_semialgebraic_set(FullSpace(), gs)   # domain for (x[1], x[2], y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "cfcf05e1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Problem\n",
      "  Name                   :                 \n",
      "  Objective sense        : minimize        \n",
      "  Type                   : CONIC (conic optimization problem)\n",
      "  Constraints            : 89              \n",
      "  Affine conic cons.     : 0               \n",
      "  Disjunctive cons.      : 0               \n",
      "  Cones                  : 0               \n",
      "  Scalar variables       : 1               \n",
      "  Matrix variables       : 7 (scalarized: 630)\n",
      "  Integer variables      : 0               \n",
      "\n",
      "Optimizer started.\n",
      "Presolve started.\n",
      "Linear dependency checker started.\n",
      "Linear dependency checker terminated.\n",
      "Eliminator started.\n",
      "Freed constraints in eliminator : 0\n",
      "Eliminator terminated.\n",
      "Eliminator - tries                  : 1                 time                   : 0.00            \n",
      "Lin. dep.  - tries                  : 1                 time                   : 0.00            \n",
      "Lin. dep.  - primal attempts        : 1                 successes              : 1               \n",
      "Lin. dep.  - dual attempts          : 0                 successes              : 0               \n",
      "Lin. dep.  - primal deps.           : 0                 dual deps.             : 0               \n",
      "Presolve terminated. Time: 0.00    \n",
      "Optimizer  - threads                : 8               \n",
      "Optimizer  - solved problem         : the primal      \n",
      "Optimizer  - Constraints            : 89              \n",
      "Optimizer  - Cones                  : 1               \n",
      "Optimizer  - Scalar variables       : 2                 conic                  : 2               \n",
      "Optimizer  - Semi-definite variables: 7                 scalarized             : 630             \n",
      "Factor     - setup time             : 0.00            \n",
      "Factor     - dense det. time        : 0.00              GP order time          : 0.00            \n",
      "Factor     - nonzeros before factor : 3924              after factor           : 4005            \n",
      "Factor     - dense dim.             : 0                 flops                  : 6.09e+05        \n",
      "ITE PFEAS    DFEAS    GFEAS    PRSTATUS   POBJ              DOBJ              MU       TIME  \n",
      "0   4.4e+02  1.0e+00  1.0e+00  0.00e+00   0.000000000e+00   0.000000000e+00   1.0e+00  0.00  \n",
      "1   2.9e+02  6.5e-01  8.1e-01  -1.01e+00  1.715226653e+00   2.271556949e+00   6.5e-01  0.00  \n",
      "2   1.9e+02  4.2e-01  6.5e-01  -1.01e+00  1.877705458e+01   2.017395223e+01   4.2e-01  0.01  \n",
      "3   2.9e+01  6.6e-02  2.6e-01  -1.00e+00  1.495305757e+02   1.635823227e+02   6.6e-02  0.01  \n",
      "4   1.0e+01  2.3e-02  1.0e-01  -7.01e-01  2.914384746e+02   3.116546264e+02   2.3e-02  0.01  \n",
      "5   5.2e+00  1.2e-02  2.1e-02  9.11e-01   1.460020672e+02   1.487939197e+02   1.2e-02  0.01  \n",
      "6   1.4e+00  3.1e-03  2.5e-03  9.66e-01   1.172184401e+02   1.178251898e+02   3.1e-03  0.02  \n",
      "7   2.1e-01  4.7e-04  1.9e-04  9.77e-01   1.192206663e+02   1.193831887e+02   4.7e-04  0.02  \n",
      "8   1.7e-02  3.7e-05  4.5e-06  9.97e-01   1.181075386e+02   1.181211196e+02   3.7e-05  0.03  \n",
      "9   7.0e-04  1.6e-06  3.9e-08  9.98e-01   1.180049990e+02   1.180055756e+02   1.6e-06  0.03  \n",
      "10  1.1e-05  2.4e-08  7.5e-11  1.00e+00   1.180000841e+02   1.180000931e+02   2.4e-08  0.03  \n",
      "11  6.4e-08  1.4e-10  3.4e-14  1.00e+00   1.180000006e+02   1.180000006e+02   1.4e-10  0.03  \n",
      "Optimizer terminated. Time: 0.03    \n",
      "\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "A JuMP Model\n",
       "├ solver: Mosek\n",
       "├ objective_sense: MIN_SENSE\n",
       "│ └ objective_function_type: VariableRef\n",
       "├ num_variables: 1\n",
       "├ num_constraints: 1\n",
       "│ └ Vector{AffExpr} in SumOfSquares.SOSPolynomialSet{BasicSemialgebraicSet{Rational{BigInt}, Polynomial{DynamicPolynomials.Commutative{DynamicPolynomials.CreationOrder}, Graded{LexOrder}, Rational{BigInt}}, AlgebraicSet{Rational{BigInt}, Polynomial{DynamicPolynomials.Commutative{DynamicPolynomials.CreationOrder}, Graded{LexOrder}, Rational{BigInt}}, Buchberger, SemialgebraicSets.SolverUsingMultiplicationMatrices{SemialgebraicSets.GröbnerBasisMultiplicationMatricesAlgorithm, ReorderedSchurMultiplicationMatricesSolver{Int64, Random.TaskLocalRNG}}, BigFloat}}, DynamicPolynomials.Monomial{DynamicPolynomials.Commutative{DynamicPolynomials.CreationOrder}, Graded{LexOrder}}, MonomialVector{DynamicPolynomials.Commutative{DynamicPolynomials.CreationOrder}, Graded{LexOrder}}, SumOfSquares.Certificate.Putinar{SumOfSquares.Certificate.MaxDegree{SOSCone, MonomialBasis}, SumOfSquares.Certificate.MaxDegree{SOSCone, MonomialBasis}}}: 1\n",
       "└ Names registered in the model\n",
       "  └ :cref, :t"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "example4 = checkSOSMonotone([[x;], [y;]], [u, -u], gs, []; specify_degree=false)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "432ccb30",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "solution_summary(; result = 1, verbose = false)\n",
      "├ solver_name          : Mosek\n",
      "├ Termination\n",
      "│ ├ termination_status : OPTIMAL\n",
      "│ ├ result_count       : 1\n",
      "│ ├ raw_status         : Mosek.MSK_SOL_STA_OPTIMAL\n",
      "│ └ objective_bound    : 1.18000e+02\n",
      "├ Solution (result = 1)\n",
      "│ ├ primal_status        : FEASIBLE_POINT\n",
      "│ ├ dual_status          : FEASIBLE_POINT\n",
      "│ ├ objective_value      : 1.18000e+02\n",
      "│ ├ dual_objective_value : 1.18000e+02\n",
      "│ └ relative_gap         : 4.44681e-10\n",
      "└ Work counters\n",
      "  ├ solve_time (sec)   : 2.70362e-02\n",
      "  ├ simplex_iterations : 0\n",
      "  ├ barrier_iterations : 11\n",
      "  └ node_count         : 0\n"
     ]
    }
   ],
   "source": [
    "# check the solution summary\n",
    "println(solution_summary(example4))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "71fac9b4",
   "metadata": {},
   "source": [
    "## Search for the closest SOS-monotone game"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "5f73489f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "payoff_basis = DynamicPolynomials.Monomial{DynamicPolynomials.Commutative{DynamicPolynomials.CreationOrder}, Graded{LexOrder}}[1, y, x, xy, x², x²y, x³, x³y, x⁴, x⁴y]\n"
     ]
    }
   ],
   "source": [
    "payoff_basis = monomials(u)  # monomial basis appearring in u\n",
    "println(\"payoff_basis = \", payoff_basis)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "f29427fa",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Algebraic Set defined by 1 equalitty\n",
       " 1//1 - z[2]^2 - z[1]^2 = 0\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# create auxiliary polynomial variables for quadratic Jacobian and define the domain\n",
    "@polyvar z[1:2]                     # auxiliary polynomial variables for quadratic Hessian/Jacobian z'Sz\n",
    "Sz = algebraic_set([1 - z'*z])      # unit sephere for the domain of z"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "b44c6fc7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "A JuMP Model\n",
       "├ solver: Mosek\n",
       "├ objective_sense: FEASIBILITY_SENSE\n",
       "├ num_variables: 0\n",
       "├ num_constraints: 0\n",
       "└ Names registered in the model: none"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Create an SOS optimization model \n",
    "model_linfnorm = SOSModel(Mosek.Optimizer)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "b0dc9898",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "p = Polynomial{DynamicPolynomials.Commutative{DynamicPolynomials.CreationOrder}, Graded{LexOrder}, VariableRef}[(_[1]) + (_[2])y + (_[3])x + (_[4])xy + (_[5])x² + (_[6])x²y + (_[7])x³ + (_[8])x³y + (_[9])x⁴ + (_[10])x⁴y]\n"
     ]
    }
   ],
   "source": [
    "# create a variable for the payoff function of P1\n",
    "p = @variable(model_linfnorm, [1:1], Poly(payoff_basis)) # payoff for P1\n",
    "println(\"p = \", p)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "a4723904",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "2×2 Matrix{Polynomial{DynamicPolynomials.Commutative{DynamicPolynomials.CreationOrder}, Graded{LexOrder}, AffExpr}}:\n",
       " (2 _[5]) + (2 _[6])y + (6 _[7])x + (6 _[8])xy + (12 _[9])x² + (12 _[10])x²y  …  (_[4]) + (2 _[6])x + (3 _[8])x² + (4 _[10])x³\n",
       " (-_[4]) + (-2 _[6])x + (-3 _[8])x² + (-4 _[10])x³                               0"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# construct the Jacobian\n",
    "vs = [differentiate(only(p), x); differentiate(only(-p), y)]  # pseudogradient vector\n",
    "jacobian = differentiate(vs, [x; y]) # Jacobian matrix\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "5cde74c8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$$ (-2 {\\_}_{5})z_{1}^{2} + (-2 {\\_}_{6})yz_{1}^{2} + (-6 {\\_}_{7})xz_{1}^{2} + (-6 {\\_}_{8})xyz_{1}^{2} + (-12 {\\_}_{9})x^{2}z_{1}^{2} + (-12 {\\_}_{10})x^{2}yz_{1}^{2} \\text{ is SOS} $$"
      ],
      "text/plain": [
       "(-2 _[5])z₁² + (-2 _[6])yz₁² + (-6 _[7])xz₁² + (-6 _[8])xyz₁² + (-12 _[9])x²z₁² + (-12 _[10])x²yz₁² is SOS"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# SOS-monotonicity constraint. The target game is SOS-monotone\n",
    "cref = @constraint(model_linfnorm, - z' * jacobian * z >= 0, domain=intersect(Sx, Sz))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "284b1bf2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$ {\\_}_{11} $"
      ],
      "text/plain": [
       "_[11]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# minimize the l-infinity distance between the new game and the original game\n",
    "t = @variable(model_linfnorm)               # t is the distance betwen the target game and the original game\n",
    "# Find the difference between the target game and the original game\n",
    "diff = vcat(coefficients.([p; -p] - [u; -u])...)\n",
    "@constraint(model_linfnorm, diff .<= t)     # linearize the l-infinity norm constraint\n",
    "@constraint(model_linfnorm, diff .>= -t)    # linearize the l-infinity norm constraint\n",
    "\n",
    "@objective(model_linfnorm, Min, t)          # minimize the distance between the new game and the original game"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "a13d5c05",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Problem\n",
      "  Name                   :                 \n",
      "  Objective sense        : minimize        \n",
      "  Type                   : CONIC (conic optimization problem)\n",
      "  Constraints            : 129             \n",
      "  Affine conic cons.     : 0               \n",
      "  Disjunctive cons.      : 0               \n",
      "  Cones                  : 0               \n",
      "  Scalar variables       : 11              \n",
      "  Matrix variables       : 7 (scalarized: 630)\n",
      "  Integer variables      : 0               \n",
      "\n",
      "Optimizer started.\n",
      "Presolve started.\n",
      "Linear dependency checker started.\n",
      "Linear dependency checker terminated.\n",
      "Eliminator started.\n",
      "Freed constraints in eliminator : 7\n",
      "Eliminator terminated.\n",
      "Eliminator - tries                  : 1                 time                   : 0.00            \n",
      "Lin. dep.  - tries                  : 1                 time                   : 0.00            \n",
      "Lin. dep.  - primal attempts        : 1                 successes              : 1               \n",
      "Lin. dep.  - dual attempts          : 0                 successes              : 0               \n",
      "Lin. dep.  - primal deps.           : 0                 dual deps.             : 0               \n",
      "Presolve terminated. Time: 0.00    \n",
      "Optimizer  - threads                : 8               \n",
      "Optimizer  - solved problem         : the primal      \n",
      "Optimizer  - Constraints            : 94              \n",
      "Optimizer  - Cones                  : 0               \n",
      "Optimizer  - Scalar variables       : 12                conic                  : 0               \n",
      "Optimizer  - Semi-definite variables: 7                 scalarized             : 630             \n",
      "Factor     - setup time             : 0.00            \n",
      "Factor     - dense det. time        : 0.00              GP order time          : 0.00            \n",
      "Factor     - nonzeros before factor : 3999              after factor           : 4080            \n",
      "Factor     - dense dim.             : 0                 flops                  : 6.11e+05        \n",
      "ITE PFEAS    DFEAS    GFEAS    PRSTATUS   POBJ              DOBJ              MU       TIME  \n",
      "0   8.9e+01  1.0e+00  1.0e+00  0.00e+00   0.000000000e+00   0.000000000e+00   1.0e+00  0.00  \n",
      "1   3.7e+01  4.2e-01  6.5e-01  -1.01e+00  1.523434480e+00   2.947602127e+00   4.2e-01  0.00  \n",
      "2   9.4e+00  1.1e-01  3.5e-01  -1.04e+00  1.321720687e+01   2.282248362e+01   1.1e-01  0.00  \n",
      "3   1.8e+00  2.1e-02  1.9e-01  -1.18e+00  1.144009885e+02   2.007099437e+02   2.1e-02  0.00  \n",
      "4   5.1e-01  5.7e-03  5.4e-02  -9.74e-01  1.930156706e+02   2.820094969e+02   5.7e-03  0.01  \n",
      "5   2.2e-01  2.5e-03  3.6e-03  2.43e+00   9.579470036e+01   9.774823024e+01   2.5e-03  0.01  \n",
      "6   5.8e-02  6.6e-04  6.3e-04  2.29e+00   5.591289327e+01   5.680725486e+01   6.6e-04  0.01  \n",
      "7   7.0e-03  7.9e-05  2.6e-05  1.10e+00   4.962836679e+01   4.973536200e+01   7.9e-05  0.01  \n",
      "8   1.8e-05  2.0e-07  3.3e-09  1.01e+00   4.900138738e+01   4.900164862e+01   2.0e-07  0.01  \n",
      "9   3.5e-07  4.0e-09  8.7e-12  1.01e+00   4.900003725e+01   4.900004185e+01   4.0e-09  0.01  \n",
      "10  1.7e-08  2.1e-10  8.5e-14  1.02e+00   4.900000200e+01   4.900000220e+01   1.9e-10  0.01  \n",
      "Optimizer terminated. Time: 0.01    \n",
      "\n"
     ]
    }
   ],
   "source": [
    "optimize!(model_linfnorm)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "7f28a4e6",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "solution_summary(; result = 1, verbose = false)\n",
      "├ solver_name          : Mosek\n",
      "├ Termination\n",
      "│ ├ termination_status : OPTIMAL\n",
      "│ ├ result_count       : 1\n",
      "│ ├ raw_status         : Mosek.MSK_SOL_STA_OPTIMAL\n",
      "│ └ objective_bound    : 4.90000e+01\n",
      "├ Solution (result = 1)\n",
      "│ ├ primal_status        : FEASIBLE_POINT\n",
      "│ ├ dual_status          : FEASIBLE_POINT\n",
      "│ ├ objective_value      : 4.90000e+01\n",
      "│ ├ dual_objective_value : 4.90000e+01\n",
      "│ └ relative_gap         : 4.08344e-09\n",
      "└ Work counters\n",
      "  ├ solve_time (sec)   : 9.07493e-03\n",
      "  ├ simplex_iterations : 0\n",
      "  ├ barrier_iterations : 10\n",
      "  └ node_count         : 0\n"
     ]
    }
   ],
   "source": [
    "# check the solution summary\n",
    "println(solution_summary(model_linfnorm))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "5149a11e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "f = -46.00000199527389 - 57.00000199527389*y - 68.00000199527389*x - 4.000001995273891*x*y - 2.19505309928536e-6*x^2 - 75.10616653801891*x^2*y - 22.88106479496601*x^3 + 32.780984089669104*x^3*y - 5.975202377011875*x^4 - 5.551923445025025*x^4*y\n",
      "u = 3 - 8*y - 19*x + 45*x*y + 49*x^2 - 89*x^2*y - 59*x^3 + 74*x^3*y + 25*x^4 - 16*x^4*y\n"
     ]
    }
   ],
   "source": [
    "# compare the original and new payoff functions\n",
    "f = only(value.(p))     # the new payoff function for P1\n",
    "println(\"f = \", f)\n",
    "println(\"u = \", u)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "54105cb9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "distance = 49.00000199527389\n"
     ]
    }
   ],
   "source": [
    "# check the minimum distance\n",
    "println(\"distance = \", value(t))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Julia 1.11.4",
   "language": "julia",
   "name": "julia-1.11"
  },
  "language_info": {
   "file_extension": ".jl",
   "mimetype": "application/julia",
   "name": "julia",
   "version": "1.11.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
