Component-Based Modeling of a Spring-Mass System
In this tutorial, we will build a simple component-based model of a spring-mass system. A spring-mass system consists of one or more masses connected by springs. Hooke's law gives the force exerted by a spring when it is extended or compressed by a given distance. This specifies a differential-equation system where the acceleration of the masses is specified using the forces acting on them.
Copy-Paste Example
using ModelingToolkit, Plots, OrdinaryDiffEq, LinearAlgebra
using ModelingToolkit: t_nounits as t, D_nounits as D
using Symbolics: scalarize
function Mass(; name, m = 1.0, xy = [0.0, 0.0], u = [0.0, 0.0])
    ps = @parameters m = m
    sts = @variables pos(t)[1:2]=xy v(t)[1:2]=u
    eqs = scalarize(D.(pos) .~ v)
    ODESystem(eqs, t, [pos..., v...], ps; name)
end
function Spring(; name, k = 1e4, l = 1.0)
    ps = @parameters k=k l=l
    @variables x(t), dir(t)[1:2]
    ODESystem(Equation[], t, [x, dir...], ps; name)
end
function connect_spring(spring, a, b)
    [spring.x ~ norm(scalarize(a .- b))
     scalarize(spring.dir .~ scalarize(a .- b))]
end
function spring_force(spring)
    -spring.k .* scalarize(spring.dir) .* (spring.x - spring.l) ./ spring.x
end
m = 1.0
xy = [1.0, -1.0]
k = 1e4
l = 1.0
center = [0.0, 0.0]
g = [0.0, -9.81]
@named mass = Mass(m = m, xy = xy)
@named spring = Spring(k = k, l = l)
eqs = [connect_spring(spring, mass.pos, center)
       scalarize(D.(mass.v) .~ spring_force(spring) / mass.m .+ g)]
@named _model = ODESystem(eqs, t, [spring.x; spring.dir; mass.pos], [])
@named model = compose(_model, mass, spring)
sys = structural_simplify(model)
prob = ODEProblem(sys, [], (0.0, 3.0))
sol = solve(prob, Rosenbrock23())
plot(sol)Explanation
Building the components
For each component, we use a Julia function that returns an ODESystem. At the top, we define the fundamental properties of a Mass: it has a mass m, a position pos and a velocity v. We also define that the velocity is the rate of change of position with respect to time.
function Mass(; name, m = 1.0, xy = [0.0, 0.0], u = [0.0, 0.0])
    ps = @parameters m = m
    sts = @variables pos(t)[1:2]=xy v(t)[1:2]=u
    eqs = scalarize(D.(pos) .~ v)
    ODESystem(eqs, t, [pos..., v...], ps; name)
endMass (generic function with 1 method)Note that this is an incompletely specified ODESystem. It cannot be simulated on its own, since the equations for the velocity v[1:2](t) are unknown. Notice the addition of a name keyword. This allows us to generate different masses with different names. A Mass can now be constructed as:
Mass(name = :mass1)\[ \begin{align} \frac{\mathrm{d} \mathtt{pos}\_{1}\left( t \right)}{\mathrm{d}t} &= v\_{1}\left( t \right) \\ \frac{\mathrm{d} \mathtt{pos}\_{2}\left( t \right)}{\mathrm{d}t} &= v\_{2}\left( t \right) \end{align} \]
Or using the @named helper macro
@named mass1 = Mass()\[ \begin{align} \frac{\mathrm{d} \mathtt{pos}\_{1}\left( t \right)}{\mathrm{d}t} &= v\_{1}\left( t \right) \\ \frac{\mathrm{d} \mathtt{pos}\_{2}\left( t \right)}{\mathrm{d}t} &= v\_{2}\left( t \right) \end{align} \]
Next, we build the spring component. It is characterized by the spring constant k and the length l of the spring when no force is applied to it. The state of a spring is defined by its current length and direction.
function Spring(; name, k = 1e4, l = 1.0)
    ps = @parameters k=k l=l
    @variables x(t), dir(t)[1:2]
    ODESystem(Equation[], t, [x, dir...], ps; name)
endSpring (generic function with 1 method)We now define functions that help construct the equations for a mass-spring system. First, the connect_spring function connects a spring between two positions a and b. Note that a and b can be the pos of a Mass, or just a fixed position such as [0., 0.]. In that sense, the length of the spring x is given by the length of the vector dir joining a and b.
function connect_spring(spring, a, b)
    [spring.x ~ norm(scalarize(a .- b))
     scalarize(spring.dir .~ scalarize(a .- b))]
endconnect_spring (generic function with 1 method)Lastly, we define the spring_force function that takes a spring and returns the force exerted by this spring.
function spring_force(spring)
    -spring.k .* scalarize(spring.dir) .* (spring.x - spring.l) ./ spring.x
endspring_force (generic function with 1 method)To create our system, we will first create the components: a mass and a spring. This is done as follows:
m = 1.0
xy = [1.0, -1.0]
k = 1e4
l = 1.0
center = [0.0, 0.0]
g = [0.0, -9.81]
@named mass = Mass(m = m, xy = xy)
@named spring = Spring(k = k, l = l)\[ \begin{align} \end{align} \]
We can now create the equations describing this system, by connecting spring to mass and a fixed point.
eqs = [connect_spring(spring, mass.pos, center)
       scalarize(D.(mass.v) .~ spring_force(spring) / mass.m .+ g)]\[ \begin{align} \mathtt{spring.x}\left( t \right) &= \sqrt{\left|\mathtt{mass.pos}\_{2}\left( t \right)\right|^{2} + \left|\mathtt{mass.pos}\_{1}\left( t \right)\right|^{2}} \\ \mathtt{spring.dir}\_{1}\left( t \right) &= \mathtt{mass.pos}\_{1}\left( t \right) \\ \mathtt{spring.dir}\_{2}\left( t \right) &= \mathtt{mass.pos}\_{2}\left( t \right) \\ \frac{\mathrm{d} \mathtt{mass.v}\_{1}\left( t \right)}{\mathrm{d}t} &= \frac{ - \mathtt{spring.dir}\_{1}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \mathtt{spring.x}\left( t \right) \right)}{\mathtt{mass.m} \mathtt{spring.x}\left( t \right)} \\ \frac{\mathrm{d} \mathtt{mass.v}\_{2}\left( t \right)}{\mathrm{d}t} &= -9.81 + \frac{ - \mathtt{spring.dir}\_{2}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \mathtt{spring.x}\left( t \right) \right)}{\mathtt{mass.m} \mathtt{spring.x}\left( t \right)} \end{align} \]
Finally, we can build the model using these equations and components.
@named _model = ODESystem(eqs, t, [spring.x; spring.dir; mass.pos], [])
@named model = compose(_model, mass, spring)\[ \begin{align} \mathtt{spring.x}\left( t \right) &= \sqrt{\left|\mathtt{mass.pos}\_{2}\left( t \right)\right|^{2} + \left|\mathtt{mass.pos}\_{1}\left( t \right)\right|^{2}} \\ \mathtt{spring.dir}\_{1}\left( t \right) &= \mathtt{mass.pos}\_{1}\left( t \right) \\ \mathtt{spring.dir}\_{2}\left( t \right) &= \mathtt{mass.pos}\_{2}\left( t \right) \\ \frac{\mathrm{d} \mathtt{mass.v}\_{1}\left( t \right)}{\mathrm{d}t} &= \frac{ - \mathtt{spring.dir}\_{1}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \mathtt{spring.x}\left( t \right) \right)}{\mathtt{mass.m} \mathtt{spring.x}\left( t \right)} \\ \frac{\mathrm{d} \mathtt{mass.v}\_{2}\left( t \right)}{\mathrm{d}t} &= -9.81 + \frac{ - \mathtt{spring.dir}\_{2}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \mathtt{spring.x}\left( t \right) \right)}{\mathtt{mass.m} \mathtt{spring.x}\left( t \right)} \\ \frac{\mathrm{d} \mathtt{mass.pos}\_{1}\left( t \right)}{\mathrm{d}t} &= \mathtt{mass.v}\_{1}\left( t \right) \\ \frac{\mathrm{d} \mathtt{mass.pos}\_{2}\left( t \right)}{\mathrm{d}t} &= \mathtt{mass.v}\_{2}\left( t \right) \end{align} \]
We can take a look at the equations in the model using the equations function.
equations(model)\[ \begin{align} \mathtt{spring.x}\left( t \right) &= \sqrt{\left|\mathtt{mass.pos}\_{2}\left( t \right)\right|^{2} + \left|\mathtt{mass.pos}\_{1}\left( t \right)\right|^{2}} \\ \mathtt{spring.dir}\_{1}\left( t \right) &= \mathtt{mass.pos}\_{1}\left( t \right) \\ \mathtt{spring.dir}\_{2}\left( t \right) &= \mathtt{mass.pos}\_{2}\left( t \right) \\ \frac{\mathrm{d} \mathtt{mass.v}\_{1}\left( t \right)}{\mathrm{d}t} &= \frac{ - \mathtt{spring.dir}\_{1}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \mathtt{spring.x}\left( t \right) \right)}{\mathtt{mass.m} \mathtt{spring.x}\left( t \right)} \\ \frac{\mathrm{d} \mathtt{mass.v}\_{2}\left( t \right)}{\mathrm{d}t} &= -9.81 + \frac{ - \mathtt{spring.dir}\_{2}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \mathtt{spring.x}\left( t \right) \right)}{\mathtt{mass.m} \mathtt{spring.x}\left( t \right)} \\ \frac{\mathrm{d} \mathtt{mass.pos}\_{1}\left( t \right)}{\mathrm{d}t} &= \mathtt{mass.v}\_{1}\left( t \right) \\ \frac{\mathrm{d} \mathtt{mass.pos}\_{2}\left( t \right)}{\mathrm{d}t} &= \mathtt{mass.v}\_{2}\left( t \right) \end{align} \]
The unknowns of this model are:
unknowns(model)7-element Vector{SymbolicUtils.BasicSymbolic{Real}}:
 spring₊x(t)
 (spring₊dir(t))[1]
 (spring₊dir(t))[2]
 (mass₊pos(t))[1]
 (mass₊pos(t))[2]
 (mass₊v(t))[1]
 (mass₊v(t))[2]And the parameters of this model are:
parameters(model)3-element Vector{Any}:
 mass₊m
 spring₊k
 spring₊lSimplifying and solving this system
This system can be solved directly as a DAE using one of the DAE solvers from DifferentialEquations.jl. However, we can symbolically simplify the system first beforehand. Running structural_simplify eliminates unnecessary variables from the model to give the leanest numerical representation of the system.
sys = structural_simplify(model)
equations(sys)\[ \begin{align} \frac{\mathrm{d} \mathtt{mass.pos}\_{1}\left( t \right)}{\mathrm{d}t} &= \mathtt{mass.v}\_{1}\left( t \right) \\ \frac{\mathrm{d} \mathtt{mass.pos}\_{2}\left( t \right)}{\mathrm{d}t} &= \mathtt{mass.v}\_{2}\left( t \right) \\ \frac{\mathrm{d} \mathtt{mass.v}\_{1}\left( t \right)}{\mathrm{d}t} &= \frac{ - \mathtt{spring.dir}\_{1}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \mathtt{spring.x}\left( t \right) \right)}{\mathtt{mass.m} \mathtt{spring.x}\left( t \right)} \\ \frac{\mathrm{d} \mathtt{mass.v}\_{2}\left( t \right)}{\mathrm{d}t} &= -9.81 + \frac{ - \mathtt{spring.dir}\_{2}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \mathtt{spring.x}\left( t \right) \right)}{\mathtt{mass.m} \mathtt{spring.x}\left( t \right)} \end{align} \]
We are left with only 4 equations involving 4 unknown variables (mass.pos[1], mass.pos[2], mass.v[1], mass.v[2]). We can solve the system by converting it to an ODEProblem. Some observed variables are not expanded by default. To view the complete equations, one can do
full_equations(sys)\[ \begin{align} \frac{\mathrm{d} \mathtt{mass.pos}\_{1}\left( t \right)}{\mathrm{d}t} &= \mathtt{mass.v}\_{1}\left( t \right) \\ \frac{\mathrm{d} \mathtt{mass.pos}\_{2}\left( t \right)}{\mathrm{d}t} &= \mathtt{mass.v}\_{2}\left( t \right) \\ \frac{\mathrm{d} \mathtt{mass.v}\_{1}\left( t \right)}{\mathrm{d}t} &= \frac{ - \mathtt{mass.pos}\_{1}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \sqrt{\left|\mathtt{mass.pos}\_{2}\left( t \right)\right|^{2} + \left|\mathtt{mass.pos}\_{1}\left( t \right)\right|^{2}} \right)}{\mathtt{mass.m} \sqrt{\left|\mathtt{mass.pos}\_{2}\left( t \right)\right|^{2} + \left|\mathtt{mass.pos}\_{1}\left( t \right)\right|^{2}}} \\ \frac{\mathrm{d} \mathtt{mass.v}\_{2}\left( t \right)}{\mathrm{d}t} &= -9.81 + \frac{ - \mathtt{mass.pos}\_{2}\left( t \right) \mathtt{spring.k} \left( - \mathtt{spring.l} + \sqrt{\left|\mathtt{mass.pos}\_{2}\left( t \right)\right|^{2} + \left|\mathtt{mass.pos}\_{1}\left( t \right)\right|^{2}} \right)}{\mathtt{mass.m} \sqrt{\left|\mathtt{mass.pos}\_{2}\left( t \right)\right|^{2} + \left|\mathtt{mass.pos}\_{1}\left( t \right)\right|^{2}}} \end{align} \]
This is done as follows:
prob = ODEProblem(sys, [], (0.0, 3.0))
sol = solve(prob, Rosenbrock23())
plot(sol)What if we want the timeseries of a different variable? That information is not lost! Instead, structural_simplify simply changes unknown variables into observed variables.
observed(sys)\[ \begin{align} \mathtt{spring.dir}\_{1}\left( t \right) &= \mathtt{mass.pos}\_{1}\left( t \right) \\ \mathtt{spring.dir}\_{2}\left( t \right) &= \mathtt{mass.pos}\_{2}\left( t \right) \\ \mathtt{spring.x}\left( t \right) &= \sqrt{\left|\mathtt{mass.pos}\_{2}\left( t \right)\right|^{2} + \left|\mathtt{mass.pos}\_{1}\left( t \right)\right|^{2}} \end{align} \]
These are explicit algebraic equations which can be used to reconstruct the required variables on the fly. This leads to dramatic computational savings since implicitly solving an ODE scales as O(n^3), so fewer unknowns are significantly better!
We can access these variables using the solution object. For example, let's retrieve the x-position of the mass over time:
sol[mass.pos[1]]1319-element Vector{Float64}:
  1.0
  0.9999999998290586
  0.999999979316097
  0.9999978938357836
  0.9999783978213862
  0.9998928643735703
  0.9995883462116874
  0.9987722296999486
  0.9968207137116937
  0.9928640064195641
  ⋮
 -0.3011467588525984
 -0.28428264096484174
 -0.2775266600735234
 -0.27584817036971526
 -0.2791160033904494
 -0.2877139453522825
 -0.3033667370847031
 -0.3259574841822946
 -0.3296931065772806We can also plot the path of the mass:
plot(sol, idxs = (mass.pos[1], mass.pos[2]))