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

Fix allocation flow output #1355

Merged
merged 3 commits into from
Apr 4, 2024
Merged
Show file tree
Hide file tree
Changes from 2 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
40 changes: 34 additions & 6 deletions core/src/allocation_optim.jl
Original file line number Diff line number Diff line change
Expand Up @@ -840,23 +840,51 @@ function save_allocation_flows!(
F_basin_out = problem[:F_basin_out]

# Edge flows
for allocation_edge in first(F.axes)
allocation_edge_idx = 1
allocation_edges = first(F.axes)
n_allocation_edges = length(allocation_edges)

while allocation_edge_idx <= n_allocation_edges
allocation_edge = allocation_edges[allocation_edge_idx]
flow_rate = JuMP.value(F[allocation_edge])

# Check whether the next allocation edge is the reverse of the current
# allocation edge
if allocation_edge_idx < n_allocation_edges &&
allocation_edges[allocation_edge_idx + 1] == reverse(allocation_edge)
# Combine the flow rates of bidirectional allocation edges
allocation_edge_idx += 1
flow_rate -= JuMP.value(F[allocation_edges[allocation_edge_idx]])
end

edge_metadata = graph[allocation_edge...]
(; node_ids) = edge_metadata

for i in eachindex(node_ids)[1:(end - 1)]
# Check in which direction this edge in the physical layer exists
if haskey(graph, node_ids[i], node_ids[i + 1])
id_from = node_ids[i]
id_to = node_ids[i + 1]
flow_rate_signed = flow_rate
else
id_from = node_ids[i + 1]
id_to = node_ids[i]
flow_rate_signed = -flow_rate
end

push!(record_flow.time, t)
push!(record_flow.edge_id, edge_metadata.id)
push!(record_flow.from_node_type, string(node_ids[i].type))
push!(record_flow.from_node_id, Int32(node_ids[i]))
push!(record_flow.to_node_type, string(node_ids[i + 1].type))
push!(record_flow.to_node_id, Int32(node_ids[i + 1]))
push!(record_flow.from_node_type, string(id_from.type))
push!(record_flow.from_node_id, Int32(id_from))
push!(record_flow.to_node_type, string(id_to.type))
push!(record_flow.to_node_id, Int32(id_to))
push!(record_flow.subnetwork_id, allocation_network_id)
push!(record_flow.priority, priority)
push!(record_flow.flow_rate, flow_rate)
push!(record_flow.flow_rate, flow_rate_signed)
push!(record_flow.optimization_type, string(optimization_type))
end

allocation_edge_idx += 1
end

# Basin flows
Expand Down
14 changes: 13 additions & 1 deletion core/test/allocation_test.jl
Original file line number Diff line number Diff line change
Expand Up @@ -202,6 +202,7 @@ end
using Ribasim: NodeID, OptimizationType
using ComponentArrays: ComponentVector
using JuMP
using DataFrames: DataFrame, ByRow, transform!

toml_path = normpath(
@__DIR__,
Expand All @@ -215,7 +216,8 @@ end
close(db)

(; allocation, user_demand, graph, basin) = p
(; allocation_models, subnetwork_demands, subnetwork_allocateds) = allocation
(; allocation_models, subnetwork_demands, subnetwork_allocateds, record_flow) =
allocation
t = 0.0

# Collecting demands
Expand Down Expand Up @@ -257,6 +259,16 @@ end

@test user_demand.allocated[2] ≈ [4.0, 0.0, 0.0]
@test user_demand.allocated[7] ≈ [0.001, 0.0, 0.0]

# Test for existence of edges in allocation flow record
allocation_flow = DataFrame(record_flow)
transform!(
allocation_flow,
[:from_node_type, :from_node_id, :to_node_type, :to_node_id] =>
ByRow((a, b, c, d) -> haskey(graph, NodeID(a, b), NodeID(c, d))) =>
:edge_exists,
)
@test all(allocation_flow.edge_exists)
end

@testitem "subnetworks with sources" begin
Expand Down
Loading