The background is I am trying to implement a bilevel optimization program using Python-based package for Adversarial Optimization (PAO), which is based on Pyomo. I am new to Pyomo, though I have used many solvers such as Gurobi and Cplex previously. To solve the bilevel optimization program, I need to use the pao.pyomo.MIBS solver. However, I found my Pyomo program can't even solve a naive unequal constraint z[0] != z[1]
correctly. My code is as follows, which is trying to assign values to z[0], z[1] \in {0, 1}
that satisfy constraint z[0] != z[1]
:
import pyomo.environ as pyo
from pyomo.environ import *
from pao.pyomo import *
N = 2
M=1024
model = ConcreteModel()
model.z = Var(range(N), domain=Integers)
for i in range(N):
model.z[i].setlb(0)
model.z[i].setub(N-1)
model.delta = Var(range(N), range(N), domain=Binary)
model.unique_z = ConstraintList()
for i in range(N):
for j in range(i+1, N):
model.unique_z.add(model.z[i] - model.z[j] + M*(1-model.delta[i, j]) >= 1)
model.unique_z.add(model.z[i] - model.z[j] - M*model.delta[i, j] <= -1)
model.obj = Objective(expr=sum(model.z[i] for i in range(N)), sense=maximize)
model.L = SubModel(fixed=model.z)
with Solver('pao.pyomo.MIBS') as solver:
results = solver.solve(model)
print("obj =", model.obj.expr(), "z =", [model.z[i].value for i in range(N)])
So the basic constraint is like here, which expresses the not equal constraint z[0] != z[1]
as a linear constraint with the help of binary variables delta
. An obvious solution is either z[0]
or z[1]
is 1 and the other variable is 0.
However, I have printed the model's output and found the model sets both z[0]
and z[1]
to be 0.
>>> model.z.pprint()
z : Size=2, Index=z_index
Key : Lower : Value : Upper : Fixed : Stale : Domain
0 : 0 : 0 : 1 : False : False : Integers
1 : 0 : 0 : 1 : False : False : Integers
>>> model.delta.pprint()
delta : Size=4, Index=delta_index
Key : Lower : Value : Upper : Fixed : Stale : Domain
(0, 0) : 0 : None : 1 : False : True : Binary
(0, 1) : 0 : 0 : 1 : False : False : Binary
(1, 0) : 0 : None : 1 : False : True : Binary
(1, 1) : 0 : None : 1 : False : True : Binary
>>> model.unique_z.pprint()
unique_z : Size=2, Index=unique_z_index, Active=True
Key : Lower : Body : Upper : Active
1 : 1.0 : z[0] - z[1] + 1024*(1 - delta[0,1]) : +Inf : True
2 : -Inf : z[0] - z[1] - 1024*delta[0,1] : -1.0 : True
So when z[0]
, z[1]
, and delta(0,1)
are 0, doesn't this violate the constraint z[0] - z[1] - 1024*delta[0,1] <= -1
? I don't understand why this can be feasible.
I'm a beginner with Pyomo. If there are any misconceptions in the model I've built or in the Pyomo outputs, I would greatly appreciate any clarification or suggestions. Thank you.
Edit: Thanks to @AirSquid, it seems the problem is with my 'pao.pyomo.MIBS' solver. Could anyone give some suggestions on what's wrong with my current implementation of 'pao.pyomo.MIBS' solver? I have to use this solver for my bilevel optimization program.
I suspect there is something wrong with your solving code and you are looking at an un-solved or erroneous solve of some kind. I'm not familiar with the code sequence you are using there. I usually:
SolverFactory
Your math is correct. I tweaked your model a tiny bit and used a different solver, but it's working. You are double importing. You should be able either use your first or second import line. I prefer the
pyo
prefix, but either way. I commented out thepao
as it isn't used in my example.Note that in
pyomo
,pprint
shows the construction w/ variable names, anddisplay
can be used to fill in (or show) the solved valuesCode:
Output: