mirror of
https://github.com/ANL-CEEESA/UnitCommitment.jl.git
synced 2025-12-07 08:48:51 -06:00
stochastic extension
This commit is contained in:
@@ -78,26 +78,25 @@ function build_model(;
|
||||
model[:obj] = AffExpr()
|
||||
model[:instance] = instance
|
||||
for g in instance.scenarios[1].units
|
||||
_add_unit_first_stage!(model, g, formulation)
|
||||
_add_unit_commitment!(model, g, formulation)
|
||||
end
|
||||
for scenario in instance.scenarios
|
||||
@info "Building scenario $(scenario.name) with" *
|
||||
"probability $(scenario.probability)"
|
||||
_setup_transmission(model, formulation.transmission, scenario)
|
||||
for l in scenario.lines
|
||||
_add_transmission_line!(model, l, formulation.transmission,
|
||||
scenario)
|
||||
for sc in instance.scenarios
|
||||
@info "Building scenario $(sc.name) with" *
|
||||
"probability $(sc.probability)"
|
||||
_setup_transmission(formulation.transmission, sc)
|
||||
for l in sc.lines
|
||||
_add_transmission_line!(model, l, formulation.transmission, sc)
|
||||
end
|
||||
for b in scenario.buses
|
||||
_add_bus!(model, b, scenario)
|
||||
for b in sc.buses
|
||||
_add_bus!(model, b, sc)
|
||||
end
|
||||
for ps in scenario.price_sensitive_loads
|
||||
_add_price_sensitive_load!(model, ps, scenario)
|
||||
for ps in sc.price_sensitive_loads
|
||||
_add_price_sensitive_load!(model, ps, sc)
|
||||
end
|
||||
for g in scenario.units
|
||||
_add_unit_second_stage!(model, g, formulation, scenario)
|
||||
for g in sc.units
|
||||
_add_unit_dispatch!(model, g, formulation, sc)
|
||||
end
|
||||
_add_system_wide_eqs!(model, scenario)
|
||||
_add_system_wide_eqs!(model, sc)
|
||||
end
|
||||
@objective(model, Min, model[:obj])
|
||||
end
|
||||
|
||||
@@ -8,7 +8,7 @@ function _add_ramp_eqs!(
|
||||
formulation_prod_vars::Gar1962.ProdVars,
|
||||
formulation_ramping::ArrCon2000.Ramping,
|
||||
formulation_status_vars::Gar1962.StatusVars,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
# TODO: Move upper case constants to model[:instance]
|
||||
RESERVES_WHEN_START_UP = true
|
||||
@@ -74,7 +74,8 @@ function _add_ramp_eqs!(
|
||||
# but the constraint below will force the unit to produce power
|
||||
eq_ramp_down[sc.name, gn, t] = @constraint(
|
||||
model,
|
||||
g.initial_power - (g.min_power[t] + prod_above[sc.name, gn, t]) <= RD
|
||||
g.initial_power -
|
||||
(g.min_power[t] + prod_above[sc.name, gn, t]) <= RD
|
||||
)
|
||||
end
|
||||
else
|
||||
|
||||
@@ -8,7 +8,7 @@ function _add_production_piecewise_linear_eqs!(
|
||||
formulation_prod_vars::Gar1962.ProdVars,
|
||||
formulation_pwl_costs::CarArr2006.PwlCosts,
|
||||
formulation_status_vars::StatusVarsFormulation,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
eq_prod_above_def = _init(model, :eq_prod_above_def)
|
||||
eq_segprod_limit = _init(model, :eq_segprod_limit)
|
||||
@@ -34,13 +34,17 @@ function _add_production_piecewise_linear_eqs!(
|
||||
|
||||
# Also add this as an explicit upper bound on segprod to make the
|
||||
# solver's work a bit easier
|
||||
set_upper_bound(segprod[sc.name, gn, t, k], g.cost_segments[k].mw[t])
|
||||
set_upper_bound(
|
||||
segprod[sc.name, gn, t, k],
|
||||
g.cost_segments[k].mw[t],
|
||||
)
|
||||
|
||||
# Definition of production
|
||||
# Equation (43) in Kneuven et al. (2020)
|
||||
eq_prod_above_def[sc.name, gn, t] = @constraint(
|
||||
model,
|
||||
prod_above[sc.name, gn, t] == sum(segprod[sc.name, gn, t, k] for k in 1:K)
|
||||
prod_above[sc.name, gn, t] ==
|
||||
sum(segprod[sc.name, gn, t, k] for k in 1:K)
|
||||
)
|
||||
|
||||
# Objective function
|
||||
|
||||
@@ -8,7 +8,7 @@ function _add_ramp_eqs!(
|
||||
formulation_prod_vars::Gar1962.ProdVars,
|
||||
formulation_ramping::DamKucRajAta2016.Ramping,
|
||||
formulation_status_vars::Gar1962.StatusVars,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
# TODO: Move upper case constants to model[:instance]
|
||||
RESERVES_WHEN_START_UP = true
|
||||
@@ -66,7 +66,8 @@ function _add_ramp_eqs!(
|
||||
elseif (t == 1 && is_initially_on) || (t > 1 && !time_invariant)
|
||||
if t > 1
|
||||
min_prod_last_period =
|
||||
prod_above[sc.name, gn, t-1] + g.min_power[t-1] * is_on[gn, t-1]
|
||||
prod_above[sc.name, gn, t-1] +
|
||||
g.min_power[t-1] * is_on[gn, t-1]
|
||||
else
|
||||
min_prod_last_period = max(g.initial_power, 0.0)
|
||||
end
|
||||
|
||||
@@ -6,7 +6,7 @@ function _add_production_vars!(
|
||||
model::JuMP.Model,
|
||||
g::Unit,
|
||||
formulation_prod_vars::Gar1962.ProdVars,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
prod_above = _init(model, :prod_above)
|
||||
segprod = _init(model, :segprod)
|
||||
@@ -23,7 +23,7 @@ function _add_production_limit_eqs!(
|
||||
model::JuMP.Model,
|
||||
g::Unit,
|
||||
formulation_prod_vars::Gar1962.ProdVars,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
eq_prod_limit = _init(model, :eq_prod_limit)
|
||||
is_on = model[:is_on]
|
||||
@@ -34,10 +34,6 @@ function _add_production_limit_eqs!(
|
||||
# Objective function terms for production costs
|
||||
# Part of (69) of Kneuven et al. (2020) as C^R_g * u_g(t) term
|
||||
|
||||
### Moving this term to another function
|
||||
# add_to_expression!(model[:obj], is_on[gn, t], g.min_power_cost[t])
|
||||
###
|
||||
|
||||
# Production limit
|
||||
# Equation (18) in Kneuven et al. (2020)
|
||||
# as \bar{p}_g(t) \le \bar{P}_g u_g(t)
|
||||
@@ -49,7 +45,8 @@ function _add_production_limit_eqs!(
|
||||
end
|
||||
eq_prod_limit[sc.name, gn, t] = @constraint(
|
||||
model,
|
||||
prod_above[sc.name, gn, t] + reserve[t] <= power_diff * is_on[gn, t]
|
||||
prod_above[sc.name, gn, t] + reserve[t] <=
|
||||
power_diff * is_on[gn, t]
|
||||
)
|
||||
end
|
||||
end
|
||||
|
||||
@@ -8,7 +8,7 @@ function _add_production_piecewise_linear_eqs!(
|
||||
formulation_prod_vars::Gar1962.ProdVars,
|
||||
formulation_pwl_costs::Gar1962.PwlCosts,
|
||||
formulation_status_vars::Gar1962.StatusVars,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
eq_prod_above_def = _init(model, :eq_prod_above_def)
|
||||
eq_segprod_limit = _init(model, :eq_segprod_limit)
|
||||
@@ -27,7 +27,8 @@ function _add_production_piecewise_linear_eqs!(
|
||||
# Equation (43) in Kneuven et al. (2020)
|
||||
eq_prod_above_def[sc.name, gn, t] = @constraint(
|
||||
model,
|
||||
prod_above[sc.name, gn, t] == sum(segprod[sc.name, gn, t, k] for k in 1:K)
|
||||
prod_above[sc.name, gn, t] ==
|
||||
sum(segprod[sc.name, gn, t, k] for k in 1:K)
|
||||
)
|
||||
|
||||
for k in 1:K
|
||||
@@ -40,12 +41,16 @@ function _add_production_piecewise_linear_eqs!(
|
||||
# that segment*
|
||||
eq_segprod_limit[sc.name, gn, t, k] = @constraint(
|
||||
model,
|
||||
segprod[sc.name, gn, t, k] <= g.cost_segments[k].mw[t] * is_on[gn, t]
|
||||
segprod[sc.name, gn, t, k] <=
|
||||
g.cost_segments[k].mw[t] * is_on[gn, t]
|
||||
)
|
||||
|
||||
# Also add this as an explicit upper bound on segprod to make the
|
||||
# solver's work a bit easier
|
||||
set_upper_bound(segprod[sc.name, gn, t, k], g.cost_segments[k].mw[t])
|
||||
set_upper_bound(
|
||||
segprod[sc.name, gn, t, k],
|
||||
g.cost_segments[k].mw[t],
|
||||
)
|
||||
|
||||
# Objective function
|
||||
# Equation (44) in Kneuven et al. (2020)
|
||||
|
||||
@@ -8,7 +8,7 @@ function _add_production_piecewise_linear_eqs!(
|
||||
formulation_prod_vars::Gar1962.ProdVars,
|
||||
formulation_pwl_costs::KnuOstWat2018.PwlCosts,
|
||||
formulation_status_vars::Gar1962.StatusVars,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
eq_prod_above_def = _init(model, :eq_prod_above_def)
|
||||
eq_segprod_limit_a = _init(model, :eq_segprod_limit_a)
|
||||
@@ -90,7 +90,8 @@ function _add_production_piecewise_linear_eqs!(
|
||||
# Equation (43) in Kneuven et al. (2020)
|
||||
eq_prod_above_def[sc.name, gn, t] = @constraint(
|
||||
model,
|
||||
prod_above[sc.name, gn, t] == sum(segprod[sc.name, gn, t, k] for k in 1:K)
|
||||
prod_above[sc.name, gn, t] ==
|
||||
sum(segprod[sc.name, gn, t, k] for k in 1:K)
|
||||
)
|
||||
|
||||
# Objective function
|
||||
@@ -103,7 +104,10 @@ function _add_production_piecewise_linear_eqs!(
|
||||
|
||||
# Also add an explicit upper bound on segprod to make the solver's
|
||||
# work a bit easier
|
||||
set_upper_bound(segprod[sc.name, gn, t, k], g.cost_segments[k].mw[t])
|
||||
set_upper_bound(
|
||||
segprod[sc.name, gn, t, k],
|
||||
g.cost_segments[k].mw[t],
|
||||
)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -8,7 +8,7 @@ function _add_ramp_eqs!(
|
||||
formulation_prod_vars::Gar1962.ProdVars,
|
||||
formulation_ramping::MorLatRam2013.Ramping,
|
||||
formulation_status_vars::Gar1962.StatusVars,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
# TODO: Move upper case constants to model[:instance]
|
||||
RESERVES_WHEN_START_UP = true
|
||||
@@ -65,7 +65,8 @@ function _add_ramp_eqs!(
|
||||
reserve[t] : 0.0
|
||||
)
|
||||
min_prod_last_period =
|
||||
g.min_power[t-1] * is_on[gn, t-1] + prod_above[sc.name, gn, t-1]
|
||||
g.min_power[t-1] * is_on[gn, t-1] +
|
||||
prod_above[sc.name, gn, t-1]
|
||||
eq_ramp_up[gn, t] = @constraint(
|
||||
model,
|
||||
max_prod_this_period - min_prod_last_period <=
|
||||
@@ -93,7 +94,8 @@ function _add_ramp_eqs!(
|
||||
# but the constraint below will force the unit to produce power
|
||||
eq_ramp_down[sc.name, gn, t] = @constraint(
|
||||
model,
|
||||
g.initial_power - (g.min_power[t] + prod_above[sc.name, gn, t]) <= RD
|
||||
g.initial_power -
|
||||
(g.min_power[t] + prod_above[sc.name, gn, t]) <= RD
|
||||
)
|
||||
end
|
||||
else
|
||||
|
||||
@@ -8,7 +8,7 @@ function _add_ramp_eqs!(
|
||||
formulation_prod_vars::Gar1962.ProdVars,
|
||||
formulation_ramping::PanGua2016.Ramping,
|
||||
formulation_status_vars::Gar1962.StatusVars,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
# TODO: Move upper case constants to model[:instance]
|
||||
RESERVES_WHEN_SHUT_DOWN = true
|
||||
@@ -68,16 +68,17 @@ function _add_ramp_eqs!(
|
||||
if UT - 2 < TRU
|
||||
# Equation (40) in Kneuven et al. (2020)
|
||||
# Covers an additional time period of the ramp-up trajectory, compared to (38)
|
||||
eq_prod_limit_ramp_up_extra_period[sc.name, gn, t] = @constraint(
|
||||
model,
|
||||
prod_above[sc.name, gn, t] +
|
||||
g.min_power[t] * is_on[gn, t] +
|
||||
reserve[t] <=
|
||||
Pbar * is_on[gn, t] - sum(
|
||||
(Pbar - (SU + i * RU)) * switch_on[gn, t-i] for
|
||||
i in 0:min(UT - 1, TRU, t - 1)
|
||||
eq_prod_limit_ramp_up_extra_period[sc.name, gn, t] =
|
||||
@constraint(
|
||||
model,
|
||||
prod_above[sc.name, gn, t] +
|
||||
g.min_power[t] * is_on[gn, t] +
|
||||
reserve[t] <=
|
||||
Pbar * is_on[gn, t] - sum(
|
||||
(Pbar - (SU + i * RU)) * switch_on[gn, t-i] for
|
||||
i in 0:min(UT - 1, TRU, t - 1)
|
||||
)
|
||||
)
|
||||
)
|
||||
end
|
||||
|
||||
# Add in shutdown trajectory if KSD >= 0 (else this is dominated by (38))
|
||||
|
||||
@@ -8,7 +8,7 @@ function _add_ramp_eqs!(
|
||||
::Gar1962.ProdVars,
|
||||
::WanHob2016.Ramping,
|
||||
::Gar1962.StatusVars,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
is_initially_on = (g.initial_status > 0)
|
||||
SU = g.startup_limit
|
||||
@@ -30,7 +30,7 @@ function _add_ramp_eqs!(
|
||||
error("Each generator may only provide one flexiramp reserve")
|
||||
end
|
||||
for r in g.reserves
|
||||
if r.type !== "up-frp" && r.type !== "down-frp"
|
||||
if r.type !== "flexiramp"
|
||||
error(
|
||||
"This formulation only supports flexiramp reserves, not $(r.type)",
|
||||
)
|
||||
@@ -39,21 +39,23 @@ function _add_ramp_eqs!(
|
||||
for t in 1:model[:instance].time
|
||||
@constraint(
|
||||
model,
|
||||
prod_above[sc.name, gn, t] + (is_on[gn, t] * minp[t]) <= mfg[sc.name, rn, gn, t]
|
||||
prod_above[sc.name, gn, t] + (is_on[gn, t] * minp[t]) <=
|
||||
mfg[sc.name, gn, t]
|
||||
) # Eq. (19) in Wang & Hobbs (2016)
|
||||
@constraint(model, mfg[sc.name, rn, gn, t] <= is_on[gn, t] * maxp[t]) # Eq. (22) in Wang & Hobbs (2016)
|
||||
@constraint(model, mfg[sc.name, gn, t] <= is_on[gn, t] * maxp[t]) # Eq. (22) in Wang & Hobbs (2016)
|
||||
if t != model[:instance].time
|
||||
@constraint(
|
||||
model,
|
||||
minp[t] * (is_on[gn, t+1] + is_on[gn, t] - 1) <=
|
||||
prod_above[sc.name, gn, t] - dwflexiramp[sc.name, rn, gn, t] +
|
||||
(is_on[gn, t] * minp[t])
|
||||
prod_above[sc.name, gn, t] -
|
||||
dwflexiramp[sc.name, rn, gn, t] + (is_on[gn, t] * minp[t])
|
||||
) # first inequality of Eq. (20) in Wang & Hobbs (2016)
|
||||
@constraint(
|
||||
model,
|
||||
prod_above[sc.name, gn, t] - dwflexiramp[sc.name, rn, gn, t] +
|
||||
prod_above[sc.name, gn, t] -
|
||||
dwflexiramp[sc.name, rn, gn, t] +
|
||||
(is_on[gn, t] * minp[t]) <=
|
||||
mfg[sc.name, rn, gn, t+1] + (maxp[t] * (1 - is_on[gn, t+1]))
|
||||
mfg[sc.name, gn, t+1] + (maxp[t] * (1 - is_on[gn, t+1]))
|
||||
) # second inequality of Eq. (20) in Wang & Hobbs (2016)
|
||||
@constraint(
|
||||
model,
|
||||
@@ -67,12 +69,12 @@ function _add_ramp_eqs!(
|
||||
prod_above[sc.name, gn, t] +
|
||||
upflexiramp[sc.name, rn, gn, t] +
|
||||
(is_on[gn, t] * minp[t]) <=
|
||||
mfg[sc.name, rn, gn, t+1] + (maxp[t] * (1 - is_on[gn, t+1]))
|
||||
mfg[sc.name, gn, t+1] + (maxp[t] * (1 - is_on[gn, t+1]))
|
||||
) # second inequality of Eq. (21) in Wang & Hobbs (2016)
|
||||
if t != 1
|
||||
@constraint(
|
||||
model,
|
||||
mfg[sc.name, rn, gn, t] <=
|
||||
mfg[sc.name, gn, t] <=
|
||||
prod_above[sc.name, gn, t-1] +
|
||||
(is_on[gn, t-1] * minp[t]) +
|
||||
(RU * is_on[gn, t-1]) +
|
||||
@@ -81,8 +83,13 @@ function _add_ramp_eqs!(
|
||||
) # Eq. (23) in Wang & Hobbs (2016)
|
||||
@constraint(
|
||||
model,
|
||||
(prod_above[sc.name, gn, t-1] + (is_on[gn, t-1] * minp[t])) -
|
||||
(prod_above[sc.name, gn, t] + (is_on[gn, t] * minp[t])) <=
|
||||
(
|
||||
prod_above[sc.name, gn, t-1] +
|
||||
(is_on[gn, t-1] * minp[t])
|
||||
) - (
|
||||
prod_above[sc.name, gn, t] +
|
||||
(is_on[gn, t] * minp[t])
|
||||
) <=
|
||||
RD * is_on[gn, t] +
|
||||
SD * (is_on[gn, t-1] - is_on[gn, t]) +
|
||||
maxp[t] * (1 - is_on[gn, t-1])
|
||||
@@ -90,7 +97,7 @@ function _add_ramp_eqs!(
|
||||
else
|
||||
@constraint(
|
||||
model,
|
||||
mfg[sc.name, rn, gn, t] <=
|
||||
mfg[sc.name, gn, t] <=
|
||||
initial_power +
|
||||
(RU * is_initially_on) +
|
||||
(SU * (is_on[gn, t] - is_initially_on)) +
|
||||
@@ -98,8 +105,10 @@ function _add_ramp_eqs!(
|
||||
) # Eq. (23) in Wang & Hobbs (2016) for the first time period
|
||||
@constraint(
|
||||
model,
|
||||
initial_power -
|
||||
(prod_above[sc.name, gn, t] + (is_on[gn, t] * minp[t])) <=
|
||||
initial_power - (
|
||||
prod_above[sc.name, gn, t] +
|
||||
(is_on[gn, t] * minp[t])
|
||||
) <=
|
||||
RD * is_on[gn, t] +
|
||||
SD * (is_initially_on - is_on[gn, t]) +
|
||||
maxp[t] * (1 - is_initially_on)
|
||||
@@ -107,7 +116,7 @@ function _add_ramp_eqs!(
|
||||
end
|
||||
@constraint(
|
||||
model,
|
||||
mfg[sc.name, rn, gn, t] <=
|
||||
mfg[sc.name, gn, t] <=
|
||||
(SD * (is_on[gn, t] - is_on[gn, t+1])) +
|
||||
(maxp[t] * is_on[gn, t+1])
|
||||
) # Eq. (24) in Wang & Hobbs (2016)
|
||||
@@ -115,7 +124,8 @@ function _add_ramp_eqs!(
|
||||
model,
|
||||
-RD * is_on[gn, t+1] -
|
||||
SD * (is_on[gn, t] - is_on[gn, t+1]) -
|
||||
maxp[t] * (1 - is_on[gn, t]) <= upflexiramp[sc.name, rn, gn, t]
|
||||
maxp[t] * (1 - is_on[gn, t]) <=
|
||||
upflexiramp[sc.name, rn, gn, t]
|
||||
) # first inequality of Eq. (26) in Wang & Hobbs (2016)
|
||||
@constraint(
|
||||
model,
|
||||
@@ -127,7 +137,8 @@ function _add_ramp_eqs!(
|
||||
@constraint(
|
||||
model,
|
||||
-RU * is_on[gn, t] - SU * (is_on[gn, t+1] - is_on[gn, t]) -
|
||||
maxp[t] * (1 - is_on[gn, t+1]) <= dwflexiramp[sc.name, rn, gn, t]
|
||||
maxp[t] * (1 - is_on[gn, t+1]) <=
|
||||
dwflexiramp[sc.name, rn, gn, t]
|
||||
) # first inequality of Eq. (27) in Wang & Hobbs (2016)
|
||||
@constraint(
|
||||
model,
|
||||
@@ -147,7 +158,8 @@ function _add_ramp_eqs!(
|
||||
) # second inequality of Eq. (28) in Wang & Hobbs (2016)
|
||||
@constraint(
|
||||
model,
|
||||
-maxp[t] * is_on[gn, t+1] <= dwflexiramp[sc.name, rn, gn, t]
|
||||
-maxp[t] * is_on[gn, t+1] <=
|
||||
dwflexiramp[sc.name, rn, gn, t]
|
||||
) # first inequality of Eq. (29) in Wang & Hobbs (2016)
|
||||
@constraint(
|
||||
model,
|
||||
@@ -157,7 +169,7 @@ function _add_ramp_eqs!(
|
||||
else
|
||||
@constraint(
|
||||
model,
|
||||
mfg[sc.name, rn, gn, t] <=
|
||||
mfg[sc.name, gn, t] <=
|
||||
prod_above[sc.name, gn, t-1] +
|
||||
(is_on[gn, t-1] * minp[t]) +
|
||||
(RU * is_on[gn, t-1]) +
|
||||
@@ -166,7 +178,10 @@ function _add_ramp_eqs!(
|
||||
) # Eq. (23) in Wang & Hobbs (2016) for the last time period
|
||||
@constraint(
|
||||
model,
|
||||
(prod_above[sc.name, gn, t-1] + (is_on[gn, t-1] * minp[t])) -
|
||||
(
|
||||
prod_above[sc.name, gn, t-1] +
|
||||
(is_on[gn, t-1] * minp[t])
|
||||
) -
|
||||
(prod_above[sc.name, gn, t] + (is_on[gn, t] * minp[t])) <=
|
||||
RD * is_on[gn, t] +
|
||||
SD * (is_on[gn, t-1] - is_on[gn, t]) +
|
||||
|
||||
@@ -2,7 +2,11 @@
|
||||
# Copyright (C) 2020, UChicago Argonne, LLC. All rights reserved.
|
||||
# Released under the modified BSD license. See COPYING.md for more details.
|
||||
|
||||
function _add_bus!(model::JuMP.Model, b::Bus, sc::UnitCommitmentScenario)::Nothing
|
||||
function _add_bus!(
|
||||
model::JuMP.Model,
|
||||
b::Bus,
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
net_injection = _init(model, :expr_net_injection)
|
||||
curtail = _init(model, :curtail)
|
||||
for t in 1:model[:instance].time
|
||||
@@ -13,7 +17,11 @@ function _add_bus!(model::JuMP.Model, b::Bus, sc::UnitCommitmentScenario)::Nothi
|
||||
curtail[sc.name, b.name, t] =
|
||||
@variable(model, lower_bound = 0, upper_bound = b.load[t])
|
||||
|
||||
add_to_expression!(net_injection[sc.name, b.name, t], curtail[sc.name, b.name, t], 1.0)
|
||||
add_to_expression!(
|
||||
net_injection[sc.name, b.name, t],
|
||||
curtail[sc.name, b.name, t],
|
||||
1.0,
|
||||
)
|
||||
add_to_expression!(
|
||||
model[:obj],
|
||||
curtail[sc.name, b.name, t],
|
||||
|
||||
@@ -6,7 +6,7 @@ function _add_transmission_line!(
|
||||
model::JuMP.Model,
|
||||
lm::TransmissionLine,
|
||||
f::ShiftFactorsFormulation,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
overflow = _init(model, :overflow)
|
||||
for t in 1:model[:instance].time
|
||||
@@ -21,30 +21,28 @@ function _add_transmission_line!(
|
||||
end
|
||||
|
||||
function _setup_transmission(
|
||||
model::JuMP.Model,
|
||||
formulation::ShiftFactorsFormulation,
|
||||
scenario::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
instance = model[:instance]
|
||||
isf = formulation.precomputed_isf
|
||||
lodf = formulation.precomputed_lodf
|
||||
if length(scenario.buses) == 1
|
||||
if length(sc.buses) == 1
|
||||
isf = zeros(0, 0)
|
||||
lodf = zeros(0, 0)
|
||||
elseif isf === nothing
|
||||
@info "Computing injection shift factors..."
|
||||
time_isf = @elapsed begin
|
||||
isf = UnitCommitment._injection_shift_factors(
|
||||
lines = scenario.lines,
|
||||
buses = scenario.buses,
|
||||
buses = sc.buses,
|
||||
lines = sc.lines,
|
||||
)
|
||||
end
|
||||
@info @sprintf("Computed ISF in %.2f seconds", time_isf)
|
||||
@info "Computing line outage factors..."
|
||||
time_lodf = @elapsed begin
|
||||
lodf = UnitCommitment._line_outage_factors(
|
||||
lines = scenario.lines,
|
||||
buses = scenario.buses,
|
||||
buses = sc.buses,
|
||||
lines = sc.lines,
|
||||
isf = isf,
|
||||
)
|
||||
end
|
||||
@@ -57,7 +55,7 @@ function _setup_transmission(
|
||||
isf[abs.(isf).<formulation.isf_cutoff] .= 0
|
||||
lodf[abs.(lodf).<formulation.lodf_cutoff] .= 0
|
||||
end
|
||||
scenario.isf = isf
|
||||
scenario.lodf = lodf
|
||||
sc.isf = isf
|
||||
sc.lodf = lodf
|
||||
return
|
||||
end
|
||||
|
||||
@@ -5,7 +5,7 @@
|
||||
function _add_price_sensitive_load!(
|
||||
model::JuMP.Model,
|
||||
ps::PriceSensitiveLoad,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
loads = _init(model, :loads)
|
||||
net_injection = _init(model, :expr_net_injection)
|
||||
@@ -15,8 +15,11 @@ function _add_price_sensitive_load!(
|
||||
@variable(model, lower_bound = 0, upper_bound = ps.demand[t])
|
||||
|
||||
# Objective function terms
|
||||
add_to_expression!(model[:obj], loads[ps.name, t],
|
||||
-ps.revenue[t] * sc.probability)
|
||||
add_to_expression!(
|
||||
model[:obj],
|
||||
loads[sc.name, ps.name, t],
|
||||
-ps.revenue[t] * sc.probability,
|
||||
)
|
||||
|
||||
# Net injection
|
||||
add_to_expression!(
|
||||
|
||||
@@ -2,22 +2,30 @@
|
||||
# Copyright (C) 2020, UChicago Argonne, LLC. All rights reserved.
|
||||
# Released under the modified BSD license. See COPYING.md for more details.
|
||||
|
||||
function _add_system_wide_eqs!(model::JuMP.Model, sc::UnitCommitmentScenario)::Nothing
|
||||
function _add_system_wide_eqs!(
|
||||
model::JuMP.Model,
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
_add_net_injection_eqs!(model, sc)
|
||||
_add_spinning_reserve_eqs!(model, sc)
|
||||
_add_flexiramp_reserve_eqs!(model, sc)
|
||||
return
|
||||
end
|
||||
|
||||
function _add_net_injection_eqs!(model::JuMP.Model, sc::UnitCommitmentScenario)::Nothing
|
||||
function _add_net_injection_eqs!(
|
||||
model::JuMP.Model,
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
T = model[:instance].time
|
||||
net_injection = _init(model, :net_injection)
|
||||
eq_net_injection = _init(model, :eq_net_injection)
|
||||
eq_power_balance = _init(model, :eq_power_balance)
|
||||
for t in 1:T, b in sc.buses
|
||||
n = net_injection[sc.name, b.name, t] = @variable(model)
|
||||
eq_net_injection[sc.name, b.name, t] =
|
||||
@constraint(model, -n + model[:expr_net_injection][sc.name, b.name, t] == 0)
|
||||
eq_net_injection[sc.name, b.name, t] = @constraint(
|
||||
model,
|
||||
-n + model[:expr_net_injection][sc.name, b.name, t] == 0
|
||||
)
|
||||
end
|
||||
for t in 1:T
|
||||
eq_power_balance[sc.name, t] = @constraint(
|
||||
@@ -28,7 +36,10 @@ function _add_net_injection_eqs!(model::JuMP.Model, sc::UnitCommitmentScenario):
|
||||
return
|
||||
end
|
||||
|
||||
function _add_spinning_reserve_eqs!(model::JuMP.Model, sc::UnitCommitmentScenario)::Nothing
|
||||
function _add_spinning_reserve_eqs!(
|
||||
model::JuMP.Model,
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
T = model[:instance].time
|
||||
eq_min_spinning_reserve = _init(model, :eq_min_spinning_reserve)
|
||||
for r in sc.reserves
|
||||
@@ -40,8 +51,11 @@ function _add_spinning_reserve_eqs!(model::JuMP.Model, sc::UnitCommitmentScenari
|
||||
# from Carrión and Arroyo (2006) and Ostrowski et al. (2012)
|
||||
eq_min_spinning_reserve[sc.name, r.name, t] = @constraint(
|
||||
model,
|
||||
sum(model[:reserve][sc.name, r.name, g.name, t] for g in r.units) +
|
||||
model[:reserve_shortfall][sc.name, r.name, t] >= r.amount[t]
|
||||
sum(
|
||||
model[:reserve][sc.name, r.name, g.name, t] for
|
||||
g in r.units
|
||||
) + model[:reserve_shortfall][sc.name, r.name, t] >=
|
||||
r.amount[t]
|
||||
)
|
||||
|
||||
# Account for shortfall contribution to objective
|
||||
@@ -57,7 +71,10 @@ function _add_spinning_reserve_eqs!(model::JuMP.Model, sc::UnitCommitmentScenari
|
||||
return
|
||||
end
|
||||
|
||||
function _add_flexiramp_reserve_eqs!(model::JuMP.Model, sc::UnitCommitmentScenario)::Nothing
|
||||
function _add_flexiramp_reserve_eqs!(
|
||||
model::JuMP.Model,
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
# Note: The flexpramp requirements in Wang & Hobbs (2016) are imposed as hard constraints
|
||||
# through Eq. (17) and Eq. (18). The constraints eq_min_upflexiramp and eq_min_dwflexiramp
|
||||
# provided below are modified versions of Eq. (17) and Eq. (18), respectively, in that
|
||||
@@ -67,39 +84,37 @@ function _add_flexiramp_reserve_eqs!(model::JuMP.Model, sc::UnitCommitmentScenar
|
||||
eq_min_dwflexiramp = _init(model, :eq_min_dwflexiramp)
|
||||
T = model[:instance].time
|
||||
for r in sc.reserves
|
||||
if r.type == "up-frp"
|
||||
for t in 1:T
|
||||
# Eq. (17) in Wang & Hobbs (2016)
|
||||
eq_min_upflexiramp[sc.name, r.name, t] = @constraint(
|
||||
model,
|
||||
sum(model[:upflexiramp][sc.name, r.name, g.name, t] for g in r.units) +
|
||||
model[:upflexiramp_shortfall][sc.name, r.name, t] >= r.amount[t]
|
||||
)
|
||||
# Account for flexiramp shortfall contribution to objective
|
||||
if r.shortfall_penalty >= 0
|
||||
add_to_expression!(
|
||||
model[:obj],
|
||||
r.shortfall_penalty * sc.probability,
|
||||
model[:upflexiramp_shortfall][sc.name, r.name, t]
|
||||
)
|
||||
end
|
||||
end
|
||||
elseif r.type == "down-frp"
|
||||
for t in 1:T
|
||||
# Eq. (18) in Wang & Hobbs (2016)
|
||||
eq_min_dwflexiramp[sc.name, r.name, t] = @constraint(
|
||||
model,
|
||||
sum(model[:dwflexiramp][sc.name, r.name, g.name, t] for g in r.units) +
|
||||
model[:dwflexiramp_shortfall][sc.name, r.name, t] >= r.amount[t]
|
||||
)
|
||||
# Account for flexiramp shortfall contribution to objective
|
||||
if r.shortfall_penalty >= 0
|
||||
add_to_expression!(
|
||||
model[:obj],
|
||||
r.shortfall_penalty * sc.probability,
|
||||
r.type == "flexiramp" || continue
|
||||
for t in 1:T
|
||||
# Eq. (17) in Wang & Hobbs (2016)
|
||||
eq_min_upflexiramp[sc.name, r.name, t] = @constraint(
|
||||
model,
|
||||
sum(
|
||||
model[:upflexiramp][sc.name, r.name, g.name, t] for
|
||||
g in r.units
|
||||
) + model[:upflexiramp_shortfall][sc.name, r.name, t] >=
|
||||
r.amount[t]
|
||||
)
|
||||
# Eq. (18) in Wang & Hobbs (2016)
|
||||
eq_min_dwflexiramp[sc.name, r.name, t] = @constraint(
|
||||
model,
|
||||
sum(
|
||||
model[:dwflexiramp][sc.name, r.name, g.name, t] for
|
||||
g in r.units
|
||||
) + model[:dwflexiramp_shortfall][sc.name, r.name, t] >=
|
||||
r.amount[t]
|
||||
)
|
||||
|
||||
# Account for flexiramp shortfall contribution to objective
|
||||
if r.shortfall_penalty >= 0
|
||||
add_to_expression!(
|
||||
model[:obj],
|
||||
r.shortfall_penalty * sc.probability,
|
||||
(
|
||||
model[:upflexiramp_shortfall][sc.name, r.name, t] +
|
||||
model[:dwflexiramp_shortfall][sc.name, r.name, t]
|
||||
)
|
||||
end
|
||||
),
|
||||
)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -2,7 +2,13 @@
|
||||
# Copyright (C) 2020, UChicago Argonne, LLC. All rights reserved.
|
||||
# Released under the modified BSD license. See COPYING.md for more details.
|
||||
|
||||
function _add_unit_first_stage!(model::JuMP.Model, g::Unit, formulation::Formulation)
|
||||
# Function for adding variables, constraints, and objective function terms
|
||||
# related to the binary commitment, startup and shutdown decisions of units
|
||||
function _add_unit_commitment!(
|
||||
model::JuMP.Model,
|
||||
g::Unit,
|
||||
formulation::Formulation,
|
||||
)
|
||||
if !all(g.must_run) && any(g.must_run)
|
||||
error("Partially must-run units are not currently supported")
|
||||
end
|
||||
@@ -21,24 +27,30 @@ function _add_unit_first_stage!(model::JuMP.Model, g::Unit, formulation::Formula
|
||||
return
|
||||
end
|
||||
|
||||
function _add_unit_second_stage!(model::JuMP.Model, g::Unit, formulation::Formulation,
|
||||
scenario::UnitCommitmentScenario)
|
||||
# Function for adding variables, constraints, and objective function terms
|
||||
# related to the continuous dispatch decisions of units
|
||||
function _add_unit_dispatch!(
|
||||
model::JuMP.Model,
|
||||
g::Unit,
|
||||
formulation::Formulation,
|
||||
sc::UnitCommitmentScenario,
|
||||
)
|
||||
|
||||
# Variables
|
||||
_add_production_vars!(model, g, formulation.prod_vars, scenario)
|
||||
_add_spinning_reserve_vars!(model, g, scenario)
|
||||
_add_flexiramp_reserve_vars!(model, g, scenario)
|
||||
_add_production_vars!(model, g, formulation.prod_vars, sc)
|
||||
_add_spinning_reserve_vars!(model, g, sc)
|
||||
_add_flexiramp_reserve_vars!(model, g, sc)
|
||||
|
||||
# Constraints and objective function
|
||||
_add_net_injection_eqs!(model, g, scenario)
|
||||
_add_production_limit_eqs!(model, g, formulation.prod_vars, scenario)
|
||||
_add_net_injection_eqs!(model, g, sc)
|
||||
_add_production_limit_eqs!(model, g, formulation.prod_vars, sc)
|
||||
_add_production_piecewise_linear_eqs!(
|
||||
model,
|
||||
g,
|
||||
formulation.prod_vars,
|
||||
formulation.pwl_costs,
|
||||
formulation.status_vars,
|
||||
scenario
|
||||
sc,
|
||||
)
|
||||
_add_ramp_eqs!(
|
||||
model,
|
||||
@@ -46,62 +58,29 @@ function _add_unit_second_stage!(model::JuMP.Model, g::Unit, formulation::Formul
|
||||
formulation.prod_vars,
|
||||
formulation.ramping,
|
||||
formulation.status_vars,
|
||||
scenario
|
||||
sc,
|
||||
)
|
||||
_add_startup_shutdown_limit_eqs!(model, g, scenario)
|
||||
_add_startup_shutdown_limit_eqs!(model, g, sc)
|
||||
return
|
||||
end
|
||||
|
||||
# function _add_unit!(model::JuMP.Model, g::Unit, formulation::Formulation)
|
||||
# if !all(g.must_run) && any(g.must_run)
|
||||
# error("Partially must-run units are not currently supported")
|
||||
# end
|
||||
# if g.initial_power === nothing || g.initial_status === nothing
|
||||
# error("Initial conditions for $(g.name) must be provided")
|
||||
# end
|
||||
|
||||
# # Variables
|
||||
# _add_production_vars!(model, g, formulation.prod_vars)
|
||||
# _add_spinning_reserve_vars!(model, g)
|
||||
# _add_flexiramp_reserve_vars!(model, g)
|
||||
# _add_startup_shutdown_vars!(model, g)
|
||||
# _add_status_vars!(model, g, formulation.status_vars)
|
||||
|
||||
# # Constraints and objective function
|
||||
# _add_min_uptime_downtime_eqs!(model, g)
|
||||
# _add_net_injection_eqs!(model, g)
|
||||
# _add_production_limit_eqs!(model, g, formulation.prod_vars)
|
||||
# _add_production_piecewise_linear_eqs!(
|
||||
# model,
|
||||
# g,
|
||||
# formulation.prod_vars,
|
||||
# formulation.pwl_costs,
|
||||
# formulation.status_vars,
|
||||
# )
|
||||
# _add_ramp_eqs!(
|
||||
# model,
|
||||
# g,
|
||||
# formulation.prod_vars,
|
||||
# formulation.ramping,
|
||||
# formulation.status_vars,
|
||||
# )
|
||||
# _add_startup_cost_eqs!(model, g, formulation.startup_costs)
|
||||
# _add_startup_shutdown_limit_eqs!(model, g)
|
||||
# _add_status_eqs!(model, g, formulation.status_vars)
|
||||
# return
|
||||
# end
|
||||
|
||||
_is_initially_on(g::Unit)::Float64 = (g.initial_status > 0 ? 1.0 : 0.0)
|
||||
|
||||
function _add_spinning_reserve_vars!(model::JuMP.Model, g::Unit, sc::UnitCommitmentScenario)::Nothing
|
||||
function _add_spinning_reserve_vars!(
|
||||
model::JuMP.Model,
|
||||
g::Unit,
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
reserve = _init(model, :reserve)
|
||||
reserve_shortfall = _init(model, :reserve_shortfall)
|
||||
for r in g.reserves
|
||||
r.type == "spinning" || continue
|
||||
for t in 1:model[:instance].time
|
||||
reserve[sc.name, r.name, g.name, t] = @variable(model, lower_bound = 0)
|
||||
reserve[sc.name, r.name, g.name, t] =
|
||||
@variable(model, lower_bound = 0)
|
||||
if (sc.name, r.name, t) ∉ keys(reserve_shortfall)
|
||||
reserve_shortfall[sc.name, r.name, t] = @variable(model, lower_bound = 0)
|
||||
reserve_shortfall[sc.name, r.name, t] =
|
||||
@variable(model, lower_bound = 0)
|
||||
if r.shortfall_penalty < 0
|
||||
set_upper_bound(reserve_shortfall[sc.name, r.name, t], 0.0)
|
||||
end
|
||||
@@ -111,35 +90,37 @@ function _add_spinning_reserve_vars!(model::JuMP.Model, g::Unit, sc::UnitCommitm
|
||||
return
|
||||
end
|
||||
|
||||
function _add_flexiramp_reserve_vars!(model::JuMP.Model, g::Unit, sc::UnitCommitmentScenario)::Nothing
|
||||
function _add_flexiramp_reserve_vars!(
|
||||
model::JuMP.Model,
|
||||
g::Unit,
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
upflexiramp = _init(model, :upflexiramp)
|
||||
upflexiramp_shortfall = _init(model, :upflexiramp_shortfall)
|
||||
mfg = _init(model, :mfg)
|
||||
dwflexiramp = _init(model, :dwflexiramp)
|
||||
dwflexiramp_shortfall = _init(model, :dwflexiramp_shortfall)
|
||||
for r in g.reserves
|
||||
if r.type == "up-frp"
|
||||
for t in 1:model[:instance].time
|
||||
# maximum feasible generation, \bar{g_{its}} in Wang & Hobbs (2016)
|
||||
mfg[sc.name, r.name, g.name, t] = @variable(model, lower_bound = 0)
|
||||
upflexiramp[sc.name, r.name, g.name, t] = @variable(model) # up-flexiramp, ur_{it} in Wang & Hobbs (2016)
|
||||
if (sc.name, r.name, t) ∉ keys(upflexiramp_shortfall)
|
||||
upflexiramp_shortfall[sc.name, r.name, t] =
|
||||
@variable(model, lower_bound = 0)
|
||||
if r.shortfall_penalty < 0
|
||||
set_upper_bound(upflexiramp_shortfall[sc.name, r.name, t], 0.0)
|
||||
end
|
||||
end
|
||||
end
|
||||
elseif r.type == "down-frp"
|
||||
for t in 1:model[:instance].time
|
||||
dwflexiramp[sc.name, r.name, g.name, t] = @variable(model) # down-flexiramp, dr_{it} in Wang & Hobbs (2016)
|
||||
if (sc.name, r.name, t) ∉ keys(dwflexiramp_shortfall)
|
||||
dwflexiramp_shortfall[sc.name, r.name, t] =
|
||||
@variable(model, lower_bound = 0)
|
||||
if r.shortfall_penalty < 0
|
||||
set_upper_bound(dwflexiramp_shortfall[sc.name, r.name, t], 0.0)
|
||||
end
|
||||
for t in 1:model[:instance].time
|
||||
# maximum feasible generation, \bar{g_{its}} in Wang & Hobbs (2016)
|
||||
mfg[sc.name, g.name, t] = @variable(model, lower_bound = 0)
|
||||
for r in g.reserves
|
||||
r.type == "flexiramp" || continue
|
||||
upflexiramp[sc.name, r.name, g.name, t] = @variable(model) # up-flexiramp, ur_{it} in Wang & Hobbs (2016)
|
||||
dwflexiramp[sc.name, r.name, g.name, t] = @variable(model) # down-flexiramp, dr_{it} in Wang & Hobbs (2016)
|
||||
if (sc.name, r.name, t) ∉ keys(upflexiramp_shortfall)
|
||||
upflexiramp_shortfall[sc.name, r.name, t] =
|
||||
@variable(model, lower_bound = 0)
|
||||
dwflexiramp_shortfall[sc.name, r.name, t] =
|
||||
@variable(model, lower_bound = 0)
|
||||
if r.shortfall_penalty < 0
|
||||
set_upper_bound(
|
||||
upflexiramp_shortfall[sc.name, r.name, t],
|
||||
0.0,
|
||||
)
|
||||
set_upper_bound(
|
||||
dwflexiramp_shortfall[sc.name, r.name, t],
|
||||
0.0,
|
||||
)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -157,7 +138,11 @@ function _add_startup_shutdown_vars!(model::JuMP.Model, g::Unit)::Nothing
|
||||
return
|
||||
end
|
||||
|
||||
function _add_startup_shutdown_limit_eqs!(model::JuMP.Model, g::Unit, sc::UnitCommitmentScenario)::Nothing
|
||||
function _add_startup_shutdown_limit_eqs!(
|
||||
model::JuMP.Model,
|
||||
g::Unit,
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
eq_shutdown_limit = _init(model, :eq_shutdown_limit)
|
||||
eq_startup_limit = _init(model, :eq_startup_limit)
|
||||
is_on = model[:is_on]
|
||||
@@ -196,7 +181,7 @@ function _add_ramp_eqs!(
|
||||
model::JuMP.Model,
|
||||
g::Unit,
|
||||
formulation::RampingFormulation,
|
||||
sc::UnitCommitmentScenario
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
prod_above = model[:prod_above]
|
||||
reserve = _total_reserves(model, g, sc)
|
||||
@@ -282,7 +267,11 @@ function _add_min_uptime_downtime_eqs!(model::JuMP.Model, g::Unit)::Nothing
|
||||
end
|
||||
end
|
||||
|
||||
function _add_net_injection_eqs!(model::JuMP.Model, g::Unit, sc::UnitCommitmentScenario)::Nothing
|
||||
function _add_net_injection_eqs!(
|
||||
model::JuMP.Model,
|
||||
g::Unit,
|
||||
sc::UnitCommitmentScenario,
|
||||
)::Nothing
|
||||
expr_net_injection = model[:expr_net_injection]
|
||||
for t in 1:model[:instance].time
|
||||
# Add to net injection expression
|
||||
@@ -305,7 +294,10 @@ function _total_reserves(model, g, sc)::Vector
|
||||
spinning_reserves = [r for r in g.reserves if r.type == "spinning"]
|
||||
if !isempty(spinning_reserves)
|
||||
reserve += [
|
||||
sum(model[:reserve][sc.name, r.name, g.name, t] for r in spinning_reserves) for t in 1:model[:instance].time
|
||||
sum(
|
||||
model[:reserve][sc.name, r.name, g.name, t] for
|
||||
r in spinning_reserves
|
||||
) for t in 1:model[:instance].time
|
||||
]
|
||||
end
|
||||
return reserve
|
||||
|
||||
Reference in New Issue
Block a user