Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Demo location routing #881

Merged
merged 7 commits into from
May 10, 2023
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
187 changes: 99 additions & 88 deletions docs/src/start/advanced-demo.jl
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ arc_costs =
locations = vcat(facilities, customers)
nb_customers = length(customers)
nb_facilities = length(facilities)
positions = 1:length(nb_positions)
positions = 1:nb_positions

# Let's define the model that will solve our problem; first we need to load some packages and dependencies:

Expand All @@ -52,12 +52,13 @@ using BlockDecomposition, Coluna
nb_routes = Int(ceil(nb_customers / nb_positions))
# We define the upper bound `nb_routes_per_locations`:
nb_routes_per_locations = min(Int(ceil(nb_routes / nb_facilities)) * 2, nb_routes)
routes_per_locations = 1:nb_routes_per_locations

# ## Direct model

# First, we solve the problem by a direct approach, using the HiGHS solver. We start by initializing the solver:

direct = JuMP.direct_model(HiGHS.Optimizer())
model = JuMP.direct_model(HiGHS.Optimizer())


# and we declare 3 types of binary variables:
Expand All @@ -66,28 +67,28 @@ direct = JuMP.direct_model(HiGHS.Optimizer())
@variable(model, z[u in locations, v in locations], Bin)
@variable(model, x[i in customers, j in facilities, k in routes_per_locations, p in positions], Bin)

# variables `y_j` indicate the opening status of a given facility ; if `y_j = 1` then facility `j` is open, otherwise `y_j = 0`
# variables `z_(u,v)` indicate which arcs are used ; if `z_(u,v) = 1` then there is an arc between location `u` and location `v`, otherwise `z_(u,v) = 0`
# variables `x_(i, j, k, p)` are used to express cover constraints and to ensure the consistency of the routes ; `x_(i, j, k, p) = 1` if customer `i` is delivered from facility `j` at the position `p` of route `k`, otherwise `x_(i, j, k, p) = 0`.
# - variables `y_j` indicate the opening status of a given facility ; if `y_j = 1` then facility `j` is open, otherwise `y_j = 0`
# - variables `z_(u,v)` indicate which arcs are used ; if `z_(u,v) = 1` then there is an arc between location `u` and location `v`, otherwise `z_(u,v) = 0`
# - variables `x_(i, j, k, p)` are used to express cover constraints and to ensure the consistency of the routes ; `x_(i, j, k, p) = 1` if customer `i` is delivered from facility `j` at the position `p` of route `k`, otherwise `x_(i, j, k, p) = 0`.

# Now, we add constraints to our model:

# "each customer is visited by at least one route"
# - "each customer is visited by at least one route"
@constraint(model, cov[i in customers],
sum(x[i, j, k, p] for j in facilities, k in routes_per_locations, p in positions) >= 1)
# "for any route from any facility, its length does not exceed the fixed maximum length `nb_positions`"
# - "for any route from any facility, its length does not exceed the fixed maximum length `nb_positions`"
@constraint(model, cardinality[j in facilities, k in routes_per_locations],
sum(x[i, j, k, p] for i in customers, p in positions) <= nb_positions * y[j])
# "only one customer can be delivered by a given route at a given position"
# - "only one customer can be delivered by a given route at a given position"
@constraint(model, assign_setup[p in positions, j in facilities, k in routes_per_locations],
sum(x[i, j, k, p] for i in customers) <= y[j])
# "a customer can only be delivered at position `p > 1` of a given route if there is a customer delivered at position `p-1` of the same route"
# - "a customer can only be delivered at position `p > 1` of a given route if there is a customer delivered at position `p-1` of the same route"
@constraint(model, open_route[j in facilities, k in routes_per_locations, p in positions; p > 1],
sum(x[i, j, k, p] for i in customers) <= sum(x[i, j, k, p-1] for i in customers))
# "there is an arc between two customers whose demand is satisfied by the same route at consecutive positions"
# - "there is an arc between two customers whose demand is satisfied by the same route at consecutive positions"
@constraint(model, route_arc[i in customers, l in customers, indi in 1:nb_customers, indl in 1:nb_customers, j in facilities, k in routes_per_locations, p in positions; p > 1 && i != l && indi != indl],
z[customers[indi], customers[indl]] >= x[l, j, k, p] + x[i, j, k, p-1] - 1)
# "there is an arc between the facility `j` and the first customer visited by the route `k` from facility `j`"
# - "there is an arc between the facility `j` and the first customer visited by the route `k` from facility `j`"
@constraint(model, start_arc[i in customers, indi in 1:nb_customers, j in facilities, k in routes_per_locations],
z[facilities[j], customers[indi]] >= x[i, j, k, 1])

Expand All @@ -96,7 +97,7 @@ direct = JuMP.direct_model(HiGHS.Optimizer())
@objective(model, Min,
sum(arc_costs[u, v] * z[u, v] for u in locations, v in locations)
+
sum(facility_fixed_costs[j] * y[j] for j in facilities))
sum(facilities_fixed_costs[j] * y[j] for j in facilities))

# ##TODO: run direct model

Expand All @@ -118,55 +119,74 @@ coluna = optimizer_with_attributes(
Coluna.Optimizer,
"params" => Coluna.Params(
solver=Coluna.Algorithm.TreeSearchAlgorithm(
maxnumnodes = 1,
maxnumnodes = 0,
conqueralg = Coluna.ColCutGenConquer(
primal_heuristics = [
##Coluna.ParameterizedHeuristic(
## Diva.Diving(),
## 1.0,
## 1.0,
## 1,
## 1,
## "Diving"
##)
]
)
) ## default branch-cut-and-price
),
"default_optimizer" => GLPK.Optimizer # GLPK for the master & the subproblems
## Coluna.ParameterizedHeuristic(
## Diva.Diving(),
## 1.0,
## 1.0,
## 1,
## 1,
## "Diving"
##)
]
)
) ## default branch-cut-and-price
),
"default_optimizer" => GLPK.Optimizer # GLPK for the master & the subproblems
)



decomp = BlockModel(coluna)
# The following method creates the model according to the decomposition described:
function create_model()

# Let's declare the variables. We distinct the master variables `y`
axis = collect(facilities)
@axis(Base_axis, axis)
@show Base_axis

@variable(decomp, y[j in facilities], Bin)

# from the sub-problem variables `x` and `z`:
model = BlockModel(coluna)

## Let's declare the variables. We distinct the master variables `y`

@variable(model, y[j in facilities], Bin)

## from the sub-problem variables `x` and `z`:

@variable(model, x[i in customers, j in Base_axis] <= 1)
@variable(model, z[u in locations, v in locations], Bin)

## The information carried by the `x` variables may seem redundant with that of the `z` variables.
## The `x` variables are in fact introduced only in order to separate a family of robust valid inequalities.

## We now declare our problem's constraints:
## The cover constraints are expressed w.r.t. the `x` variables:
@constraint(model, cov[i in customers],
sum(x[i, j] for j in facilities) >= 1)

## We add a constraint to express that if a facility is not opened, there can be no arc between this facility and a customer.
@constraint(model, open_facility[j in facilities],
sum(z[j, i] for i in customers) <= y[j] * nb_routes_per_locations)


@variable(decomp, x[i in customers, j in Base_axis] <= 1)
@variable(decomp, z[u in locations, v in locations], Bin)
## Contrary to the direct model, we are not obliged here to add constraints to ensure the consistency of the routes because we solve our subproblems by pricing. It will therefore be the responsibility of the pricing callback to create consistent routes.

# The information carried by the `x` variables may seem redundant with that of the `z` variables.
# The `x` variables are in fact introduced only in order to separate a family of robust valid inequalities.
## We set the objective function:
@objective(model, Min,
sum(arc_costs[u, v] * z[u, v] for u in locations, v in locations)
+
sum(facilities_fixed_costs[j] * y[j] for j in facilities))

# We now declare our problem's constraints:
# The cover constraints are expressed w.r.t. the `x` variables:
@constraint(decomp, cov[i in customers],
sum(x[i, j] for j in facilities) >= 1)
@dantzig_wolfe_decomposition(model, dec, Base_axis)
subproblems = BlockDecomposition.getsubproblems(dec)
specify!.(subproblems, lower_multiplicity=0, upper_multiplicity=nb_routes_per_locations, solver=my_pricing_callback)
subproblemrepresentative.(z, Ref(subproblems))

# We add a constraint to express that if a facility is not opened, there can be no arc between this facility and a customer.
@constraint(decomp, open_facility[j in facilities],
sum(z[j, i] for i in customers) <= y[j] * nb_routes_per_locations)
return model, x, y, z, cov
end

# We do not need to add constraints ensuring the consistency of the routes because we solve our subproblems by pricing. It will therefore be the responsibility of the pricing callback to create consistent routes. ##TODO link with additional constraints of Direct model

# We set the objective function:

@objective(decomp, Min,
sum(arc_costs[u, v] * z[u, v] for u in locations, v in locations)
+
sum(facilities_fixed_costs[j] * y[j] for j in facilities))

# ## Pricing callback

Expand Down Expand Up @@ -259,11 +279,13 @@ function z_contribution(enroute::EnumeratedRoute, z_red_costs)
return z
end



# We are now able to write our pricing callback:

function my_pricing_callback(cbdata)
## get the id of the facility
j = BlockDecomposition.indice(BlockDecomposition.callback_spid(cbdata, decomp))
j = BlockDecomposition.indice(BlockDecomposition.callback_spid(cbdata, model))

## retrieve variables reduced costs
z_red_costs = Dict(
Expand Down Expand Up @@ -298,23 +320,19 @@ function my_pricing_callback(cbdata)
sol_cost = min_reduced_cost

## Submit the solution of the subproblem to Coluna
MOI.submit(decomp, BlockDecomposition.PricingSolution(cbdata), sol_cost, sol_vars, sol_vals)
MOI.submit(model, BlockDecomposition.PricingSolution(cbdata), sol_cost, sol_vars, sol_vals)

## Submit the dual bound to the solution of the subproblem
## This bound is used to compute the contribution of the subproblem to the lagrangian
## bound in column generation.
MOI.submit(decomp, BlockDecomposition.PricingDualBound(cbdata), sol_cost) # optimal solution
MOI.submit(model, BlockDecomposition.PricingDualBound(cbdata), sol_cost) # optimal solution

end


# Set decomposition
#@dantzig_wolfe_decomposition(decomp, dec, Base_axis)
#subproblems = BlockDecomposition.getsubproblems(dec)
#specify!.(subproblems, lower_multiplicity=0, upper_multiplicity=nb_routes_per_locations, solver = my_pricing_callback) #set solver = nothing to test without my_pricing_callback
#subproblemrepresentative.(z, Ref(subproblems))
##Solve the model
#JuMP.optimize!(decomp)
# Create the model:
(model, x, y, z, _) = create_model()
# Solve:
JuMP.optimize!(model)

# ## TODO: display "raw" decomp model output and comment, transition to next section

Expand Down Expand Up @@ -354,28 +372,22 @@ function valid_inequalities_callback(cbdata)
## Add the valid inequalities to the model.
for ineq in inequalities
constr = JuMP.@build_constraint(x[ineq.customer_id, ineq.facility_id] <= y[ineq.facility_id])
MOI.submit(decomp, MOI.UserCut(cbdata), constr)
MOI.submit(model, MOI.UserCut(cbdata), constr)
end
end

MOI.set(decomp, MOI.UserCutCallback(), valid_inequalities_callback);

# We re-declare the model and solve it with the inequalities_callback:
(model, x, y, z, _) = create_model()
MOI.set(model, MOI.UserCutCallback(), valid_inequalities_callback);
JuMP.optimize!(model)


#MOI.set(decomp, MOI.UserCutCallback(), valid_inequalities_callback);
#@dantzig_wolfe_decomposition(decomp, dec, Base_axis)
#subproblems = BlockDecomposition.getsubproblems(dec)
#specify!.(subproblems, lower_multiplicity=0, upper_multiplicity=nb_routes_per_locations, solver = my_pricing_callback) #set solver = nothing to test without my_pricing_callback
#subproblemrepresentative.(z, Ref(subproblems))
#JuMP.optimize!(decomp)

# ## TODO: comment on the improvement of the dual bound, fix display of both raw decomp and valid ineq decomp model
# ## TODO: comment on the improvement of the dual bound



# ## Strengthen with non-robust cuts (rank-one cuts)

# ##TODO: describe the goal by looking at lambdas values -> highlight that the aim is to drive them towards integrality
# ##TODO: describe the goal by looking at lambdas values (if possible?) -> highlight that the aim is to drive them towards integrality

# Here, we implement special types of cuts called "rank-one cuts" (R1C). These cuts are non-robust in the sense that they can not be expressed only with the original variables of the model. In particular, they have to be expressed with the master columns variables `λ_k`.
# R1Cs are obtained by applying the Chvátal-Gomory procedure once, hence their name, on cover constraints. We must therefore be able to differentiate the cover constraints from the other constraints of the model. To do this, we exploit an advantage of Coluna that allows us to attach custom data to the constraints and variables of our model:
Expand All @@ -384,8 +396,11 @@ MOI.set(decomp, MOI.UserCutCallback(), valid_inequalities_callback);
struct CoverConstrData <: BlockDecomposition.AbstractCustomData
customer::Int
end

(model, x, y, z, cov) = create_model()

# We declare our custom data to Coluna
BlockDecomposition.customconstrs!(decomp, CoverConstrData);
BlockDecomposition.customconstrs!(model, CoverConstrData);
# And we attach one custom data to each cover constraint
for i in customers
customdata!(cov[i], CoverConstrData(i))
Expand All @@ -394,7 +409,7 @@ end

# The rank-one cuts we are going to add are of the form:
# `sum(c_k λ_k) <= 1.0`
# for a fixed subset `r1c_cov_constrs` of cover constraints of size 3, with `λ_k` the master columns variables and c_k` s.t.
# for a fixed subset `r1c_cov_constrs` of cover constraints of size 3, with `λ_k` the master columns variables and `c_k` s.t.
# `c_k = ⌊ 1/2 x |r1c_locations ∩ r1c_cov_constrs| ⌋`
# with `r1c_locations` the current solution (route) that corresponds to `λ_k`.
# e.g. if we consider cover constraints cov[3], cov[6] and cov[8] in our cut, then the route 1-4-6-7 gives a zero coefficient while the route 1-4-6-3 gives a coefficient equal to one.
Expand All @@ -412,8 +427,8 @@ struct R1cCutData <: BlockDecomposition.AbstractCustomData
end

# We declare our custom data to Coluna:
BlockDecomposition.customvars!(decomp, R1cVarData)
BlockDecomposition.customconstrs!(decomp, [CoverConstrData, R1cCutData]);
BlockDecomposition.customvars!(model, R1cVarData)
BlockDecomposition.customconstrs!(model, [CoverConstrData, R1cCutData]);

# This method is called by Coluna to compute the coefficients of the `λ_k` in the cuts:
function Coluna.MathProg.computecoeff(
Expand Down Expand Up @@ -464,7 +479,7 @@ function r1c_callback(cbdata)
end
if violation > 1
## create the constraint and add it to the model, use custom data to keep information about the cut (= the subset of considered cover constraints)
MOI.submit(decomp,
MOI.submit(model,
MOI.UserCut(cbdata),
JuMP.ScalarConstraint(JuMP.AffExpr(0.0), MOI.LessThan(1.0)),
R1cCutData(cov_constr_subset)
Expand All @@ -490,7 +505,7 @@ end

# We re-write our pricing callback, with the additional contribution that corresponds to R1Cs cost:
function my_pricing_callback(cbdata)
j = BlockDecomposition.indice(BlockDecomposition.callback_spid(cbdata, decomp))
j = BlockDecomposition.indice(BlockDecomposition.callback_spid(cbdata, model))
z_red_costs = Dict(
"z_$(u)_$(v)" => BlockDecomposition.callback_reduced_cost(cbdata, z[u, v]) for u in locations, v in locations
)
Expand Down Expand Up @@ -529,20 +544,16 @@ function my_pricing_callback(cbdata)
z_vars = [z[u, v] for (u,v) in best_route_arcs]
x_vars = [x[i, j] for i in best_route_customers]
sol_vars = vcat(z_vars, x_vars)
sol_vals = vcat([1.0 for _ in z_vars], [1.0 for _ in x_vars])
sol_vals = ones(Float64, length(z_vars) + length(x_vars))
sol_cost = min_reduced_cost

## Submit the solution of the subproblem to Coluna
## TODO: comment on custom data here
MOI.submit(decomp, BlockDecomposition.PricingSolution(cbdata), sol_cost, sol_vars, sol_vals, R1cVarData(best_route.path))
MOI.submit(decomp, BlockDecomposition.PricingDualBound(cbdata), sol_cost)
MOI.submit(model, BlockDecomposition.PricingSolution(cbdata), sol_cost, sol_vars, sol_vals, R1cVarData(best_route.path))
MOI.submit(model, BlockDecomposition.PricingDualBound(cbdata), sol_cost)

end


MOI.set(decomp, MOI.UserCutCallback(), r1c_callback);
@dantzig_wolfe_decomposition(decomp, dec, Base_axis)
subproblems = BlockDecomposition.getsubproblems(dec)
specify!.(subproblems, lower_multiplicity=0, upper_multiplicity=nb_routes_per_locations, solver = my_pricing_callback)
subproblemrepresentative.(z, Ref(subproblems))
JuMP.optimize!(decomp)
MOI.set(model, MOI.UserCutCallback(), r1c_callback);
JuMP.optimize!(model)