Tutorial on Stepwise Initialization of a Complex Model

This example demonstrates how to initialize a complex network model with both static and dynamic components. The models are closely related to the ones used in the gas network example, but greatly simplified for the sake of this tutorial. We'll create a gas network model with three nodes and pipes connecting them, and show how to:

  1. Create static models for initialization
  2. Find a steady-state solution
  3. Create corresponding dynamic models
  4. Initialize the dynamic models with the steady-state solution
  5. Simulate the system with dynamic behavior

This script can be downloaded as a normal Julia script here.

First, let's import the necessary packages:

using NetworkDynamics
using ModelingToolkit
using ModelingToolkit: t_nounits as t, D_nounits as D
using OrdinaryDiffEqTsit5
using CairoMakie

Node Models

We'll start by defining our node models using ModelingToolkit. First, let's create a template for common states and equations in all gas nodes:

@mtkmodel GasNode begin
    @variables begin
        p(t), [description="Pressure"] # node output
        q̃_nw(t), [description="aggregated flow from pipes into node"] # node input
        q̃_inj(t), [description="flow injected into the network"]
    end
    @equations begin
        q̃_inj ~ -q̃_nw
    end
end

Now we'll define three specific node types:

A) A constant pressure node that forces pressure to maintain a specific value

@mtkmodel ConstantPressureNode begin
    @extend GasNode()
    @parameters begin
        p_set, [description="Constant pressure setpoint"]
    end
    @equations begin
        p ~ p_set
    end
end

B) A static prosumer node which forces a certain flow (pressure is fully implicit)

@mtkmodel StaticProsumerNode begin
    @extend GasNode()
    @parameters begin
        q̃_prosumer, [description="flow injected by prosumer"]
    end
    @equations begin
        -q̃_nw ~ q̃_prosumer
    end
end

C) A dynamic prosumer node with compliance, which adds dynamics to the pressure state

@mtkmodel DynamicProsumerNode begin
    @extend GasNode()
    @parameters begin
        q̃_prosumer, [description="flow injected by prosumer"]
        C=0.1, [description="Compliance"]
    end
    @equations begin
        C*D(p) ~ q̃_prosumer + q̃_nw
    end
end

D) A pressure control node that tries to maintain a set pressure by adjusting its injection

@mtkmodel PressureControlNode begin
    @extend GasNode()
    @parameters begin
        p_set, [description="Pressure setpoint", guess=1]
        K_p=1, [description="Proportional gain"]
        K_i=1, [description="Integral gain"]
        C=0.1, [description="Compliance"]
    end
    @variables begin
        Δp(t), [description="Pressure error"]
        ξ(t), [description="Integral state", guess=0]
        q̃_prosumer(t), [description="flow injected by producer"]
    end
    @equations begin
        Δp ~ p_set - p
        D(ξ) ~ Δp
        q̃_prosumer ~ K_p*Δp + K_i*ξ
        C*D(p) ~ q̃_prosumer + q̃_nw
    end
end

Edge Models

Now we'll define our edge models, starting with a template for the pipe:

@mtkmodel GasPipe begin
    @variables begin
        q̃(t), [description="flow through pipe"] #output
        p_src(t), [description="pressure at start of pipe"] #input
        p_dst(t), [description="pressure at end of pipe"] #input
    end
end

Next, we define a dynamic pipe with inertia (a simple delayed model):

@mtkmodel DynamicPipe begin
    @extend GasPipe()
    @parameters begin
        R=0.1, [description="Resistance"]
        M=0.1, [description="Inertia"]
    end
    @equations begin
        M*D(q̃) ~ (p_src - p_dst)/R - q̃ # some simple delayed model
    end
end

And finally a quasistatic pipe model for initialization purposes. This equals the dynamic model in steady state, making it ideal for finding initial conditions:

@mtkmodel QuasistaticPipe begin
    @extend GasPipe()
    @parameters begin
        R=0.1, [description="Resistance"]
    end
    @equations begin
        q̃ ~ (p_src - p_dst)/R
    end
end

Defining a Static Model for Initialization

Our first step is to define a static model that we'll use to find the steady-state solution. This is a crucial step for initializing complex dynamic models.

Step 1: Define all the components of our static model First, node 1 is our producer which will later be a controlled producer. For initialization, we use a static model:

@named v1_mod_static = ConstantPressureNode(p_set=1)
v1_static = VertexModel(v1_mod_static, [:q̃_nw], [:p], vidx=1)

# Nodes 2 and 3 are consumers. For them, we'll use static prosumer models:
@named v2_mod_static = StaticProsumerNode(q̃_prosumer=-0.6) # consumer
v2_static = VertexModel(v2_mod_static, [:q̃_nw], [:p], vidx=2)

@named v3_mod_static = StaticProsumerNode(q̃_prosumer=-0.4) # consumer
v3_static = VertexModel(v3_mod_static, [:q̃_nw], [:p], vidx=3)

Now we define the static pipe models connecting our nodes:

@named p_mod_static = QuasistaticPipe()
p12_static = EdgeModel(p_mod_static, [:p_src], [:p_dst], AntiSymmetric([:q̃]), src=1, dst=2)
p13_static = EdgeModel(p_mod_static, [:p_src], [:p_dst], AntiSymmetric([:q̃]), src=1, dst=3)
p23_static = EdgeModel(p_mod_static, [:p_src], [:p_dst], AntiSymmetric([:q̃]), src=2, dst=3)

Assemble all components into a static network:

nw_static = Network([v1_static, v2_static, v3_static], [p12_static, p13_static, p23_static])
Network with 2 states and 6 parameters
 ├─ 3 vertices (2 unique types)
 └─ 3 edges (1 unique type)
Edge-Aggregation using SequentialAggregator(+)

Create an initial guess for the steady state and modify it with reasonable values:

u_static_guess = NWState(nw_static)
u_static_guess.v[2, :p] = 1.0
u_static_guess.v[3, :p] = 1.0

Find the steady-state solution using our initial guess:

u_static = find_fixpoint(nw_static, u_static_guess)
NWState{Vector{Float64}} of Network (3 vertices, 3 edges)
  ├─ VIndex(2, :p) => 0.9466666666666667
  └─ VIndex(3, :p) => 0.9533333333333334
 p = NWParameter([1.0, -0.6, -0.4, 0.1, 0.1, 0.1])
 t = nothing

Defining a Dynamic Model

Now we'll define our dynamic model using more complex components:

@named v1_mod_dyn = PressureControlNode()
v1_dyn = VertexModel(v1_mod_dyn, [:q̃_nw], [:p], vidx=1)

@named v2_mod_dyn = DynamicProsumerNode(q̃_prosumer=-0.6)
v2_dyn = VertexModel(v2_mod_dyn, [:q̃_nw], [:p], vidx=2)

@named v3_mod_dyn = DynamicProsumerNode(q̃_prosumer=-0.4)
v3_dyn = VertexModel(v3_mod_dyn, [:q̃_nw], [:p], vidx=3)

Create dynamic pipe models with inertia:

@named p_mod_dyn = DynamicPipe()
p12_dyn = EdgeModel(p_mod_dyn, [:p_src], [:p_dst], AntiSymmetric([:q̃]), src=1, dst=2)
p13_dyn = EdgeModel(p_mod_dyn, [:p_src], [:p_dst], AntiSymmetric([:q̃]), src=1, dst=3)
p23_dyn = EdgeModel(p_mod_dyn, [:p_src], [:p_dst], AntiSymmetric([:q̃]), src=2, dst=3)

Assemble the dynamic network:

nw_dyn = Network([v1_dyn, v2_dyn, v3_dyn], [p12_dyn, p13_dyn, p23_dyn])
Network with 7 states and 14 parameters
 ├─ 3 vertices (2 unique types)
 └─ 3 edges (1 unique type)
Edge-Aggregation using SequentialAggregator(+)

Initializing the Dynamic Model with the Static Solution

Now comes the important part: we need to initialize the interface values (pressures and flows) of the dynamic model with the results from the static model.

We can do this manually:

# Vertex 1: output
set_default!(nw_dyn[VIndex(1)], :p, u_static.v[1, :p])
# Vertex 1: input
set_default!(nw_dyn[VIndex(1)], :q̃_nw, u_static.v[1, :q̃_nw])

But there is also a built-in method set_interface_defaults! which we can use automatically:

set_interface_defaults!(nw_dyn, u_static; verbose=true)
Setting the interface defaults:
  Vertex 1 output: p => 1.0
  Vertex 1 input:  q̃_nw => -0.9999999999999998
  Vertex 2 output: p => 0.9466666666666667
  Vertex 2 input:  q̃_nw => 0.6000000000000005
  Vertex 3 output: p => 0.9533333333333334
  Vertex 3 input:  q̃_nw => 0.39999999999999925
  Edge 1 output: ₋q̃ => -0.5333333333333334
  Edge 1 output: q̃ => 0.5333333333333334
  Edge 1 input:  p_src => 1.0
  Edge 1 input:  p_dst => 0.9466666666666667
  Edge 2 output: ₋q̃ => -0.46666666666666634
  Edge 2 output: q̃ => 0.46666666666666634
  Edge 2 input:  p_src => 1.0
  Edge 2 input:  p_dst => 0.9533333333333334
  Edge 3 output: ₋q̃ => 0.0666666666666671
  Edge 3 output: q̃ => -0.0666666666666671
  Edge 3 input:  p_src => 0.9466666666666667
  Edge 3 input:  p_dst => 0.9533333333333334

With the interfaces all set, we can "initialize" the internal states of the dynamic models.

For example, let's inspect the state of our first vertex:

nw_dyn[VIndex(1)]
VertexModel :v1_mod_dyn PureStateMap() @ Vertex 1
 ├─ 1 input:  [q̃_nw=-1]
 ├─ 2 states: [ξ≈0, p=1]
 ├─ 1 output: [p=1]
 └─ 4 params: [K_i=1, C=0.1, K_p=1, p_set≈1]

We observe that both the initial state ξ as well as the pressure setpoint p_set are left "free". Using initialize_component!, we can try to find values for the "free" states and parameters such that the interface constraints are fulfilled.

initialize_component!(nw_dyn[VIndex(1)])
VertexModel :v1_mod_dyn PureStateMap() @ Vertex 1
 ├─ 1 input:  [q̃_nw=-1]
 ├─ 2 states: [ξ=1, p=1]
 ├─ 1 output: [p=1]
 └─ 4 params: [K_i=1, C=0.1, K_p=1, p_set=1]

We may also use dump_initial_state to get a more detailed view of the state:

dump_initial_state(nw_dyn[VIndex(1)])
Inputs:
  q̃_nw        = -1
States:
  p           =  1
  ξ           =  1   (guess  0)
Outputs:
  p           =  1
Parameters:
  C           =  0.1
  K_i         =  1
  K_p         =  1
  p_set       =  1   (guess  1)
Observed:
  q̃_inj       =  1
  q̃_prosumer  =  1
  Δp          =  0

We can also initialize the other two vertices, however it is unnecessary since their state is already completely determined by the fixed input/output:

initialize_component!(nw_dyn[VIndex(2)])
initialize_component!(nw_dyn[VIndex(3)])
[ Info: No free variables! Residual 5.551115123125783e-15
[ Info: No free variables! Residual 7.771561172376096e-15

Similarly, we can initialize the dynamic pipe models. However, since their dynamic state equals the output, once again there is nothing to initialize.

initialize_component!(nw_dyn[EIndex(1)])
initialize_component!(nw_dyn[EIndex(2)])
initialize_component!(nw_dyn[EIndex(3)])
[ Info: No free variables! Residual 0.0
[ Info: No free variables! Residual 0.0
[ Info: No free variables! Residual 0.0

Now, everything is initialized, which means every input, output, state and parameter either has a default metadata or an init metadata. When constructing the NWState for this network, it will be filled with all those values which should now correspond to a steady state of the system:

u0_dyn = NWState(nw_dyn)
NWState{Vector{Float64}} of Network (3 vertices, 3 edges)
  ├─ VIndex(1, :ξ)  => 0.9999999999999997
  ├─ VIndex(1, :p)  => 1.0
  ├─ VIndex(2, :p)  => 0.9466666666666667
  ├─ VIndex(3, :p)  => 0.9533333333333334
  ├─ EIndex(1, :q̃)  => 0.5333333333333334
  ├─ EIndex(2, :q̃)  => 0.46666666666666634
  └─ EIndex(3, :q̃)  => -0.0666666666666671
 p = NWParameter([1.0, 0.1, 1.0, 1.0, 0.1, -0.6, 0.1, -0.4, 0.1, 0.1, 0.1, 0.1, 0.1, 0.1])
 t = nothing

Let's verify that our initialization is correct by checking that the derivatives are close to zero:

du = ones(dim(nw_dyn))
nw_dyn(du, uflat(u0_dyn), pflat(u0_dyn), 0.0)
extrema(du .- zeros(dim(nw_dyn))) # very close to zero, confirming we have a steady state!
(-7.771561172376096e-15, 5.551115123125783e-15)

Simulating the Dynamic Model

Now we can solve the dynamic model and add a disturbance to see how the system responds:

affect = ComponentAffect([], [:q̃_prosumer]) do u, p, ctx
    @info "Increase consumer demand at t=$(ctx.t)"
    p[:q̃_prosumer] -= 0.1
end
cb = PresetTimeComponentCallback([1.0], affect)
set_callback!(nw_dyn[VIndex(2)], cb) # attach disturbance to second node

Create and solve the ODE problem with the callback:

prob = ODEProblem(nw_dyn, copy(uflat(u0_dyn)), (0, 7), copy(pflat(u0_dyn));
    callback=get_callbacks(nw_dyn))
sol = solve(prob, Tsit5())
[ Info: Increase consumer demand at t=1.0

Visualizing the Results

Finally, let's visualize the results of our simulation. The plots show how our gas network responds to the increased consumer demand at t=1:

  1. Pressure at nodes: We see a pressure drop at all nodes after the disturbance before the pressure is stabilized by the controller.

  2. Injection by producer: Node 1 increases its injection to compensate for the higher demand.

  3. Draw by consumers: The solid lines show the actual flows at nodes 2 and 3, while the dashed lines show the set consumer demands. At t=1, we see the step change in consumer demand at node 2.

  4. Flows through pipes: Shows how the flows in all pipes adjust to the new demand pattern.

let
    fig = Figure(size=(1000,1000))
    ax = Axis(fig[1, 1]; title="Pressure at nodes")
    for i in 1:3
        lines!(ax, sol; idxs=VIndex(i, :p), label="Node $i", color=Cycled(i))
    end

    ax = Axis(fig[2, 1]; title="Injection by producer")
    lines!(ax, sol; idxs=VIndex(1, :q̃_inj), label="Node 1", color=Cycled(1))

    ax = Axis(fig[3, 1]; title="Draw by consumers")
    for i in 2:3
        lines!(ax, sol; idxs=@obsex(-1*VIndex(i, :q̃_inj)), label="Node $i", color=Cycled(i))
        lines!(ax, sol; idxs=@obsex(-1*VIndex(i, :q̃_prosumer)), label="Node $i", linestyle=:dash, color=Cycled(i))
    end

    ax = Axis(fig[4, 1]; title="Flows through pipes")
    for i in 1:3
        lines!(ax, sol; idxs=@obsex(abs(EIndex(i, :q̃))), label="Pipe $i", color=Cycled(i))
    end

    fig
end
Example block output

This page was generated using Literate.jl.