Initialization

Initialization of the system describes the process of finding valid initial conditions, mostly a fixpoint of the system. We distinguish between two types of initialization: full system initialization and component initialization.

Full-System Initialization

Full system initialization describes the process of finding a fixpoint/steady state of the entire system.

To do so, you can use find_fixpoint, which creates a SteadyStateProblem of the whole network and tries to solve it.

Component-wise Initialization

In contrast to full-system initialization the goal of component-wise initialization is to find a valid initial condition for a single component first, given a network coupling.

This can be useful in cases, where there are nontrivial internal dynamics and states within a single vertex or edge. The idea of component-wise initialization is to find internal states which match a given "network coupling" (fixed inputs and outputs).

The following initialization workflow (as used in the Tutorial on Initialization) is quite common for complex dynamics:

  1. Define simple, quasistatic models with the same input-output structure as the dynamic models.
  2. Find a solution of the static model using find_fixpoint.
  3. Define elaborate, dynamical models. Define dynamical network on same graph topology.
  4. Use set_interface_defaults! to "copy" the inputs and outputs of all components from the static solution to the dynamical network.
  5. Use initialize_component! to fix "free" states and parameters within the dynamical models to find a steady state while satisfying the constraints on inputs/output.

Let's consider the following example of a Swing-equation generator model.

using NetworkDynamics, ModelingToolkit
using ModelingToolkit: t_nounits as t, D_nounits as Dt

@mtkmodel Swing begin
    @variables begin
        u_r(t)=1, [description="bus d-voltage", output=true]
        u_i(t)=0.1, [description="bus q-voltage", output=true]
        i_r(t)=1, [description="bus d-current (flowing into bus)", input=true]
        i_i(t)=0.1, [description="bus d-current (flowing into bus)", input=true]
        ω(t), [guess=0.0, description="Rotor frequency"]
        θ(t), [guess=0.0, description="Rotor angle"]
        Pel(t), [guess=1, description="Electrical Power injected into the grid"]
    end
    @parameters begin
        M=0.005, [description="Inertia"]
        D=0.1, [description="Damping"]
        V=sqrt(u_r^2 + u_i^2), [description="Voltage magnitude"]
        ω_ref=0, [description="Reference frequency"]
        Pm, [guess=0.1,description="Mechanical Power"]
    end
    @equations begin
        Dt(θ) ~ ω - ω_ref
        Dt(ω) ~ 1/M * (Pm - D*ω - Pel)
        Pel ~ u_r*i_r + u_i*i_i
        u_r ~ V*cos(θ)
        u_i ~ V*sin(θ)
    end
end
sys = Swing(name=:swing)
vf = VertexModel(sys, [:i_r, :i_i], [:u_r, :u_i])
VertexModel :swing NoFeedForward()
 ├─ 2 inputs:  [i_r=1, i_i=0.1]
 ├─ 2 states:  [θ≈0, ω≈0]
 ├─ 2 outputs: [u_r=1, u_i=0.1]
 └─ 5 params:  [M=0.005, Pm≈0.1, V=1.005, D=0.1, ω_ref=0]

You can see in the provided metadata, that we've set default values for the node outputs u_r, u_i, the node inputs i_r, i_i and most parameters. For some states and parameters, we've only provided a guess rather than a default. Variables which only have guesses are considered "tunable" for the initialization algorithm.

In order to initialize the remaining variables we use initialize_component!, which is a mutating function which tries to solve the nonlinear initialization problem and store the found values for the "free" variables as init metadata.

initialize_component!(vf; verbose=true)
[ Info: Initialization problem is overconstrained (3 vars for 4 equations). Create NonlinearLeastSquaresProblem for [:θ, :ω, :Pm].
[ Info: Initialization successful with residual 1.1188630228293672e-16
VertexModel :swing NoFeedForward()
 ├─ 2 inputs:  [i_r=1, i_i=0.1]
 ├─ 2 states:  [θ=0.099669, ω=-8.8849e-24]
 ├─ 2 outputs: [u_r=1, u_i=0.1]
 └─ 5 params:  [M=0.005, Pm=1.01, V=1.005, D=0.1, ω_ref=0]

Which led to a successful initialization of states and as well as parameter :Pm. To retrieve the residual you can use init_residual.

As a quick test we can ensure that the angle indeed matches the voltage angle:

get_init(vf, :θ) ≈ atan(get_default(vf, :u_i), get_default(vf, :u_r))
true

It is possible to inspect initial states (also for observed symbols) using get_initial_state. You can print out the whole state using dump_initial_state.

dump_initial_state(vf)
Inputs:
  i_i   =  0.1
  i_r   =  1
States:
  θ     =  0.099669   (guess  0)
  ω     = -8.8849e-24 (guess  0)
Outputs:
  u_i   =  0.1
  u_r   =  1
Parameters:
  D     =  0.1
  M     =  0.005
  Pm    =  1.01       (guess  0.1)
  V     =  1.005
  ω_ref =  0
Observed:
  Pel   =  1.01