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

[DNMY] Support in the loop PowerFlow evaluation #1040

Draft
wants to merge 45 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
1c30601
add PFS deps
jd-lara Jan 9, 2024
0bcb68e
make time_steps consistent
jd-lara Jan 9, 2024
cac31f2
add PowerFlows
jd-lara Jan 9, 2024
1b7d415
add new aux variables for power flow
jd-lara Jan 9, 2024
02c52ed
make time steps consistent
jd-lara Jan 9, 2024
1e71c06
fix failing test
jd-lara Jan 9, 2024
a94defb
add power flow evaluator to model and container
jd-lara Jan 9, 2024
d1336c0
add evaluation in initialization
jd-lara Jan 9, 2024
7d1b0d2
add code for evaluation
jd-lara Jan 9, 2024
c4071f4
add missing kwarg
jd-lara Jan 9, 2024
61ee4b5
fix incorrect var type
jd-lara Jan 9, 2024
3b31d02
fix old bug
jd-lara Jan 9, 2024
347b67c
update aux vars for power flow evaluation
jd-lara Jan 9, 2024
5606f65
WIP: function for the power flow evaluation
jd-lara Jan 9, 2024
e8caaca
whitespace
jd-lara Jan 9, 2024
198e7b4
Merge branch 'main' into jd/pf_integration
jd-lara Jan 23, 2024
bf872ca
bump deps
jd-lara Jan 23, 2024
4e0f8ee
change to varnames
jd-lara Jan 23, 2024
7a2555a
add auxiliary variables
jd-lara Jan 23, 2024
d95ef9e
add wrapper for power flow data
jd-lara Jan 24, 2024
d2b26ac
use wrapper
jd-lara Jan 24, 2024
ddb92e3
add powerflow wrapper
jd-lara Jan 24, 2024
33bd977
power flow rename
jd-lara Jan 24, 2024
2645ced
file rename
jd-lara Jan 24, 2024
c997e2f
add evaluation to the container
jd-lara Jan 25, 2024
1ba76cb
WIP: add functions to map results to pf
jd-lara Jan 25, 2024
c11f4cf
Merge branch 'psy4' into jd/pf_integration
jd-lara Mar 6, 2024
4fcfa16
Merge branch 'main' into jd/pf_integration
GabrielKS Aug 19, 2024
b26e2b0
Remove redundant code
GabrielKS Oct 15, 2024
54e4859
Rewrite power flow evaluation to support more types of power flow
GabrielKS Oct 15, 2024
1e714ac
Support evaluation of multiple power flows in the loop
GabrielKS Oct 15, 2024
508db25
Better accommodate `PSSEExporter` in the loop, reduce duplication
GabrielKS Oct 16, 2024
b6fc453
Trait-based special behavior for power flow aux vars
GabrielKS Oct 17, 2024
eb51cfb
Scaffold subtype-based updating of power flow aux vars
GabrielKS Oct 17, 2024
dfd7405
Update power flow aux vars from `PowerFlowData`
GabrielKS Oct 17, 2024
c0ac0be
Complete prototype power flow in the loop -> export implementation
GabrielKS Oct 17, 2024
9e1f30a
Power flow in the loop: handle load special cases
GabrielKS Oct 28, 2024
1507b14
Power flow in the loop: support multi period export
GabrielKS Oct 28, 2024
9ec1356
Misc code cleanup following self-review
GabrielKS Oct 28, 2024
ad25918
Add basic tests of power flow in the loop
GabrielKS Nov 4, 2024
958e68d
Update `DISABLED_TEST_FILES`
GabrielKS Nov 4, 2024
187f473
Merge branch 'main' into jd/pf_integration
GabrielKS Nov 4, 2024
e016521
PERF: don't use `get_bus(sys, number)`, it's O(n)
GabrielKS Nov 19, 2024
30454e6
Add aux vars, infrastructure to receive results from AC power flow
GabrielKS Jan 16, 2025
a242ddf
Fix `bus_activepower_withdrawals` sign error
GabrielKS Jan 17, 2025
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
2 changes: 2 additions & 0 deletions Project.toml
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ JuMP = "4076af6c-e467-56ae-b986-b466b2749572"
LinearAlgebra = "37e2e46d-f89d-539d-b4ee-838fcccc9c8e"
Logging = "56ddb016-857b-54e1-b83d-db4d58db5568"
MathOptInterface = "b8f27783-ece8-5eb3-8dc8-9495eed66fee"
PowerFlows = "94fada2c-fd9a-4e89-8d82-81405f5cb4f6"
PowerModels = "c36e90e8-916a-50a6-bd94-075b64ef4655"
PowerNetworkMatrices = "bed98974-b02a-5e2f-9fe0-a103f5c450dd"
PowerSystems = "bcd98974-b02a-5e2f-9ee0-a103f5c450dd"
Expand Down Expand Up @@ -47,6 +48,7 @@ Logging = "1"
MathOptInterface = "1"
PowerModels = "~0.19"
PowerNetworkMatrices = "^0.9"
PowerFlows = "0.6"
PowerSystems = "^3"
PrettyTables = "2"
ProgressMeter = "^1.5"
Expand Down
7 changes: 7 additions & 0 deletions src/PowerSimulations.jl
Original file line number Diff line number Diff line change
Expand Up @@ -239,6 +239,10 @@ export LowerBoundFeedForwardSlack
export TimeDurationOn
export TimeDurationOff
export PowerOutput
export PowerFlowAngle
export PowerFlowVoltage
export PowerFlowLineReactivePower
export PowerFlowLineActivePower

# Constraints
export AbsoluteValueConstraint
Expand Down Expand Up @@ -340,6 +344,7 @@ import LinearAlgebra
import JSON3
import PowerSystems
import InfrastructureSystems
import PowerFlows
import PowerNetworkMatrices
import PowerNetworkMatrices: PTDF, VirtualPTDF
export PTDF
Expand Down Expand Up @@ -400,6 +405,7 @@ const MOI = MathOptInterface
const MOIU = MathOptInterface.Utilities
const MOPFM = MOI.FileFormats.Model
const PNM = PowerNetworkMatrices
const PFS = PowerFlows
const TS = TimeSeries

################################################################################
Expand Down Expand Up @@ -531,6 +537,7 @@ include("network_models/pm_translator.jl")
include("network_models/network_slack_variables.jl")
include("network_models/area_balance_model.jl")
include("network_models/hvdc_networks.jl")
include("network_models/powerflow_evaluation.jl")

include("initial_conditions/initialization.jl")

Expand Down
22 changes: 22 additions & 0 deletions src/core/auxiliary_variables.jl
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,29 @@ Auxiliary Variable for Thermal Generation Models that solve for power above min
"""
struct PowerOutput <: AuxVariableType end

"""
Auxiliary Variable for the bus angle results from power flow evaluation
"""
struct PowerFlowAngle <: AuxVariableType end

"""
Auxiliary Variable for the bus voltage magnitued results from power flow evaluation
"""
struct PowerFlowVoltage <: AuxVariableType end

"""
Auxiliary Variable for the line reactive flow from power flow evaluation
"""
struct PowerFlowLineReactivePower <: AuxVariableType end

"""
Auxiliary Variable for the line active flow from power flow evaluation
"""
struct PowerFlowLineActivePower <: AuxVariableType end

should_write_resulting_value(::Type{<:AuxVariableType}) = true

convert_result_to_natural_units(::Type{<:AuxVariableType}) = false
convert_result_to_natural_units(::Type{PowerOutput}) = true
convert_result_to_natural_units(::Type{PowerFlowLineReactivePower}) = true
convert_result_to_natural_units(::Type{PowerFlowLineActivePower}) = true
4 changes: 4 additions & 0 deletions src/core/network_model.jl
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ mutable struct NetworkModel{T <: PM.AbstractPowerModel}
duals::Vector{DataType}
radial_branches::PNM.RadialBranches
reduce_radial_branches::Bool
powerflow_evaluation::Union{Nothing, PFS.PowerFlowEvaluationModel}
jd-lara marked this conversation as resolved.
Show resolved Hide resolved

function NetworkModel(
::Type{T};
Expand All @@ -42,6 +43,7 @@ mutable struct NetworkModel{T <: PM.AbstractPowerModel}
reduce_radial_branches = false,
subnetworks = Dict{Int, Set{Int}}(),
duals = Vector{DataType}(),
powerflow_evaluation = nothing,
) where {T <: PM.AbstractPowerModel}
_check_pm_formulation(T)
new{T}(
Expand All @@ -52,6 +54,7 @@ mutable struct NetworkModel{T <: PM.AbstractPowerModel}
duals,
PNM.RadialBranches(),
reduce_radial_branches,
powerflow_evaluation,
)
end
end
Expand All @@ -66,6 +69,7 @@ get_reference_buses(m::NetworkModel{T}) where {T <: PM.AbstractPowerModel} =
collect(keys(m.subnetworks))
get_subnetworks(m::NetworkModel) = m.subnetworks
get_bus_area_map(m::NetworkModel) = m.bus_area_map
get_powerflow_evaluation(m::NetworkModel) = m.powerflow_evaluation
has_subnetworks(m::NetworkModel) = !isempty(m.bus_area_map)

function add_dual!(model::NetworkModel, dual)
Expand Down
4 changes: 3 additions & 1 deletion src/core/optimization_container.jl
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,7 @@ mutable struct OptimizationContainer <: AbstractModelContainer
built_for_recurrent_solves::Bool
metadata::OptimizationContainerMetadata
default_time_series_type::Type{<:PSY.TimeSeriesData}
power_flow_data::Union{PFS.PowerFlowData, Nothing}
end

function OptimizationContainer(
Expand Down Expand Up @@ -154,6 +155,7 @@ function OptimizationContainer(
false,
OptimizationContainerMetadata(),
T,
nothing,
)
end

Expand Down Expand Up @@ -648,8 +650,8 @@ function build_impl!(
@debug "Total operation count $(PSI.get_jump_model(container).operator_counter)" _group =
LOG_GROUP_OPTIMIZATION_CONTAINER

add_power_flow_data!(container, get_powerflow_evaluation(transmission_model), sys)
check_optimization_container(container)

return
end

Expand Down
2 changes: 2 additions & 0 deletions src/initial_conditions/initialization.jl
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,8 @@ function get_initial_conditions_template(model::OperationModel)
)
network_model.radial_branches = get_radial_branches(get_network_model(model.template))
network_model.subnetworks = get_subnetworks(get_network_model(model.template))
# Initialization does not support PowerFlow evaluation
network_model.powerflow_evaluation = nothing
bus_area_map = get_bus_area_map(get_network_model(model.template))
if !isempty(bus_area_map)
network_model.bus_area_map = get_bus_area_map(get_network_model(model.template))
Expand Down
8 changes: 2 additions & 6 deletions src/network_models/network_constructor.jl
Original file line number Diff line number Diff line change
Expand Up @@ -202,39 +202,35 @@ function construct_network!(
end

if get_use_slacks(model)
add_variables!(container, SystemBalanceSlackUp, sys, T)
add_variables!(container, SystemBalanceSlackDown, sys, T)
add_variables!(container, SystemBalanceSlackUp, sys, model)
add_variables!(container, SystemBalanceSlackDown, sys, model)
add_to_expression!(
container,
ActivePowerBalance,
SystemBalanceSlackUp,
sys,
model,
T,
)
add_to_expression!(
container,
ActivePowerBalance,
SystemBalanceSlackDown,
sys,
model,
T,
)
add_to_expression!(
container,
ReactivePowerBalance,
SystemBalanceSlackUp,
sys,
model,
T,
)
add_to_expression!(
container,
ReactivePowerBalance,
SystemBalanceSlackDown,
sys,
model,
T,
)
objective_function!(container, PSY.ACBus, model)
end
Expand Down
72 changes: 72 additions & 0 deletions src/network_models/powerflow_evaluation.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
function add_power_flow_data!(::OptimizationContainer, ::Nothing, ::PSY.System)
# NO OP function
end

function _add_branches_aux_variables!(
container::OptimizationContainer,
vars::Vector{DataType},
branch_types::Vector{DataType},
branch_names::Vector{String},
)
time_steps = get_time_steps(container)
for var_type in vars
for D in Set(branch_types)
add_aux_variable_container!(
container,
var_type,
D,
branch_names[branch_types == D],
jd-lara marked this conversation as resolved.
Show resolved Hide resolved
time_steps,
)
end
end
return
end

function _add_buses_aux_variables!(
container::OptimizationContainer,
vars::Vector{DataType},
bus_names::Vector{String},
)
time_steps = get_time_steps(container)
for var_type in vars
add_aux_variable_container!(
container,
var_type,
D,
bus_names,
time_steps,
)
end
return
end

function add_power_flow_data!(
container::OptimizationContainer,
evaluator::T,
sys::PSY.System,
) where {T <: Union{PFS.PTDFDCPowerFlow, PFS.vPTDFDCPowerFlow}}
@info "Building PowerFlow evaluator using $(evaluator)"
container.power_flow_data =
PFS.PowerFlowData(evaluator, sys; time_steps = length(get_time_steps(container)))
end

function add_power_flow_data!(
container::OptimizationContainer,
evaluator::PFS.DCPowerFlow,
sys::PSY.System,
)
@info "Building PowerFlow evaluator using $(evaluator)"
container.power_flow_data =
PFS.PowerFlowData(evaluator, sys; time_steps = length(get_time_steps(container)))
end

function add_power_flow_data!(
container::OptimizationContainer,
evaluator::PFS.ACPowerFlow,
sys::PSY.System,
)
@info "Building PowerFlow evaluator using $(evaluator)"
container.power_flow_data =
PFS.PowerFlowData(evaluator, sys; time_steps = length(get_time_steps(container)))
end
8 changes: 4 additions & 4 deletions src/services_models/transmission_interface.jl
Original file line number Diff line number Diff line change
Expand Up @@ -33,26 +33,26 @@ function add_constraints!(container::OptimizationContainer,
model::ServiceModel{T, ConstantMaxInterfaceFlow},
) where {T <: PSY.TransmissionInterface}
expr = get_expression(container, InterfaceTotalFlow(), T)
interfaces, timesteps = axes(expr)
interfaces, time_steps = axes(expr)
constraint_container_ub = lazy_container_addition!(
container,
InterfaceFlowLimit(),
T,
interfaces,
timesteps;
time_steps;
meta = "ub",
)
constraint_container_lb = lazy_container_addition!(
container,
InterfaceFlowLimit(),
T,
interfaces,
timesteps;
time_steps;
meta = "lb",
)
int_name = PSY.get_name(interface)
min_flow, max_flow = PSY.get_active_power_flow_limits(interface)
for t in timesteps
for t in time_steps
constraint_container_ub[int_name, t] =
JuMP.@constraint(get_jump_model(container), expr[int_name, t] <= max_flow)
constraint_container_lb[int_name, t] =
Expand Down
6 changes: 3 additions & 3 deletions src/utils/jump_utils.jl
Original file line number Diff line number Diff line change
Expand Up @@ -106,9 +106,9 @@ function _to_matrix(
array::SparseAxisArray{T, N, K},
columns,
) where {T, N, K <: NTuple{N, Any}}
timesteps = Set{Int}(k[N] for k in keys(array.data))
data = Matrix{Float64}(undef, length(timesteps), length(columns))
for (ix, col) in enumerate(columns), t in timesteps
time_steps = Set{Int}(k[N] for k in keys(array.data))
data = Matrix{Float64}(undef, length(time_steps), length(columns))
for (ix, col) in enumerate(columns), t in time_steps
data[t, ix] = array.data[(col..., t)]
end
return data
Expand Down
1 change: 1 addition & 0 deletions test/Project.toml
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ JuMP = "4076af6c-e467-56ae-b986-b466b2749572"
Logging = "56ddb016-857b-54e1-b83d-db4d58db5568"
Memento = "f28f55f0-a522-5efc-85c2-fe41dfb9b2d9"
Pkg = "44cfe95a-1eb2-52ea-b672-e2afdf69b78f"
PowerFlows = "94fada2c-fd9a-4e89-8d82-81405f5cb4f6"
PowerModels = "c36e90e8-916a-50a6-bd94-075b64ef4655"
PowerNetworkMatrices = "bed98974-b02a-5e2f-9fe0-a103f5c450dd"
PowerSimulations = "e690365d-45e2-57bb-ac84-44ba829e73c4"
Expand Down
50 changes: 25 additions & 25 deletions test/test_network_constructors.jl
Original file line number Diff line number Diff line change
@@ -1,26 +1,27 @@
# Note to devs. Use GLPK or Cbc for models with linear constraints and linear cost functions
# Use OSQP for models with quadratic cost function and linear constraints and ipopt otherwise
const networks_for_testing = networks = [
(PM.ACPPowerModel, fast_ipopt_optimizer),
(PM.ACRPowerModel, fast_ipopt_optimizer),
(PM.ACTPowerModel, fast_ipopt_optimizer),
#(PM.IVRPowerModel, fast_ipopt_optimizer), #instantiate_ivp_expr_model not implemented
(PM.DCPPowerModel, fast_ipopt_optimizer),
(PM.DCMPPowerModel, fast_ipopt_optimizer),
(PM.NFAPowerModel, fast_ipopt_optimizer),
(PM.DCPLLPowerModel, fast_ipopt_optimizer),
(PM.LPACCPowerModel, fast_ipopt_optimizer),
(PM.SOCWRPowerModel, fast_ipopt_optimizer),
(PM.SOCWRConicPowerModel, scs_solver),
(PM.QCRMPowerModel, fast_ipopt_optimizer),
(PM.QCLSPowerModel, fast_ipopt_optimizer),
#(PM.SOCBFPowerModel, fast_ipopt_optimizer), # not implemented
(PM.BFAPowerModel, fast_ipopt_optimizer),
#(PM.SOCBFConicPowerModel, fast_ipopt_optimizer), # not implemented
(PM.SDPWRMPowerModel, scs_solver),
(PM.SparseSDPWRMPowerModel, scs_solver),
(PTDFPowerModel, fast_ipopt_optimizer),
]
const networks_for_testing =
networks = [
(PM.ACPPowerModel, fast_ipopt_optimizer),
(PM.ACRPowerModel, fast_ipopt_optimizer),
(PM.ACTPowerModel, fast_ipopt_optimizer),
#(PM.IVRPowerModel, fast_ipopt_optimizer), #instantiate_ivp_expr_model not implemented
(PM.DCPPowerModel, fast_ipopt_optimizer),
(PM.DCMPPowerModel, fast_ipopt_optimizer),
(PM.NFAPowerModel, fast_ipopt_optimizer),
(PM.DCPLLPowerModel, fast_ipopt_optimizer),
(PM.LPACCPowerModel, fast_ipopt_optimizer),
(PM.SOCWRPowerModel, fast_ipopt_optimizer),
(PM.SOCWRConicPowerModel, scs_solver),
(PM.QCRMPowerModel, fast_ipopt_optimizer),
(PM.QCLSPowerModel, fast_ipopt_optimizer),
#(PM.SOCBFPowerModel, fast_ipopt_optimizer), # not implemented
(PM.BFAPowerModel, fast_ipopt_optimizer),
#(PM.SOCBFConicPowerModel, fast_ipopt_optimizer), # not implemented
(PM.SDPWRMPowerModel, scs_solver),
(PM.SparseSDPWRMPowerModel, scs_solver),
(PTDFPowerModel, fast_ipopt_optimizer),
]

jd-lara marked this conversation as resolved.
Show resolved Hide resolved
@testset "All PowerModels models construction" begin
c_sys5 = PSB.build_system(PSITestSystems, "c_sys5")
Expand Down Expand Up @@ -871,7 +872,6 @@ end
end

@testset "DCPPowerModel Radial Branches Test" begin

net_model = DCPPowerModel

template_uc = template_unit_commitment(;
Expand Down Expand Up @@ -937,9 +937,9 @@ end
for (network, solver) in networks_for_testing
template = get_thermal_dispatch_template_network(
NetworkModel(network;
PTDF_matrix = PTDF(c_sys5),
reduce_radial_branches = true,
use_slacks),
PTDF_matrix = PTDF(c_sys5),
reduce_radial_branches = true,
use_slacks = true),
)
ps_model = DecisionModel(template, c_sys5; optimizer = solver)
@test build!(ps_model; output_dir = mktempdir(; cleanup = true)) ==
Expand Down
Loading