Skip to content

Commit fe19d26

Browse files
Merge pull request #3497 from AayushSabharwal/as/nonamespace
feat: allow toggling namespacing independently of `complete`
2 parents 44dff8e + b54b56f commit fe19d26

18 files changed

+343
-44
lines changed

docs/src/basics/AbstractSystem.md

+12
Original file line numberDiff line numberDiff line change
@@ -148,3 +148,15 @@ The `AbstractSystem` types allow for specifying default values, for example
148148
into the value maps, where for any repeats the value maps override the default.
149149
In addition, defaults of a higher level in the system override the defaults of
150150
a lower level in the system.
151+
152+
## Namespacing
153+
154+
By default, unsimplified systems will namespace variables accessed via `getproperty`.
155+
Systems created via `@mtkbuild`, or ones passed through `structural_simplify` or
156+
`complete` will not perform this namespacing. However, all of these processes modify
157+
the system in a variety of ways. To toggle namespacing without transforming any other
158+
property of the system, use `toggle_namespacing`.
159+
160+
```@docs
161+
toggle_namespacing
162+
```

src/ModelingToolkit.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -328,7 +328,7 @@ export alg_equations, diff_equations, has_alg_equations, has_diff_equations
328328
export get_alg_eqs, get_diff_eqs, has_alg_eqs, has_diff_eqs
329329

330330
export @variables, @parameters, @independent_variables, @constants, @brownian
331-
export @named, @nonamespace, @namespace, extend, compose, complete
331+
export @named, @nonamespace, @namespace, extend, compose, complete, toggle_namespacing
332332
export debug_system
333333

334334
#export ContinuousClock, Discrete, sampletime, input_timedomain, output_timedomain

src/systems/abstractsystem.jl

+36-2
Original file line numberDiff line numberDiff line change
@@ -590,7 +590,24 @@ function SymbolicIndexingInterface.all_symbols(sys::AbstractSystem)
590590
return syms
591591
end
592592

593+
"""
594+
$(TYPEDSIGNATURES)
595+
596+
Check whether a system is marked as `complete`.
597+
"""
593598
iscomplete(sys::AbstractSystem) = isdefined(sys, :complete) && getfield(sys, :complete)
599+
"""
600+
$(TYPEDSIGNATURES)
601+
602+
Check whether a system performs namespacing.
603+
"""
604+
function does_namespacing(sys::AbstractSystem)
605+
if isdefined(sys, :namespacing)
606+
getfield(sys, :namespacing)
607+
else
608+
!iscomplete(sys)
609+
end
610+
end
594611

595612
"""
596613
$(TYPEDSIGNATURES)
@@ -798,9 +815,25 @@ function complete(
798815
if isdefined(sys, :initializesystem) && get_initializesystem(sys) !== nothing
799816
@set! sys.initializesystem = complete(get_initializesystem(sys); split)
800817
end
818+
sys = toggle_namespacing(sys, false; safe = true)
801819
isdefined(sys, :complete) ? (@set! sys.complete = true) : sys
802820
end
803821

822+
"""
823+
$(TYPEDSIGNATURES)
824+
825+
Return a new `sys` with namespacing enabled or disabled, depending on `value`. The
826+
keyword argument `safe` denotes whether systems that do not support such a toggle
827+
should error or be ignored.
828+
"""
829+
function toggle_namespacing(sys::AbstractSystem, value::Bool; safe = false)
830+
if !isdefined(sys, :namespacing)
831+
safe && return sys
832+
throw(ArgumentError("The system must define the `namespacing` flag to toggle namespacing"))
833+
end
834+
@set sys.namespacing = value
835+
end
836+
804837
"""
805838
$(TYPEDSIGNATURES)
806839
@@ -985,13 +1018,14 @@ function Base.propertynames(sys::AbstractSystem; private = false)
9851018
end
9861019
end
9871020

988-
function Base.getproperty(sys::AbstractSystem, name::Symbol; namespace = !iscomplete(sys))
1021+
function Base.getproperty(
1022+
sys::AbstractSystem, name::Symbol; namespace = does_namespacing(sys))
9891023
if has_parent(sys) && (parent = get_parent(sys); parent !== nothing)
9901024
return getproperty(parent, name; namespace)
9911025
end
9921026
wrap(getvar(sys, name; namespace = namespace))
9931027
end
994-
function getvar(sys::AbstractSystem, name::Symbol; namespace = !iscomplete(sys))
1028+
function getvar(sys::AbstractSystem, name::Symbol; namespace = does_namespacing(sys))
9951029
systems = get_systems(sys)
9961030
if isdefined(sys, name)
9971031
Base.depwarn(

src/systems/analysis_points.jl

+7-1
Original file line numberDiff line numberDiff line change
@@ -768,7 +768,13 @@ already an `AnalysisPoint` or `Symbol`) is simply wrapped in an array. `Symbol`
768768
`AnalysisPoint`s are namespaced with `sys`.
769769
"""
770770
canonicalize_ap(sys::AbstractSystem, ap::Symbol) = [AnalysisPoint(renamespace(sys, ap))]
771-
canonicalize_ap(sys::AbstractSystem, ap::AnalysisPoint) = [ap]
771+
function canonicalize_ap(sys::AbstractSystem, ap::AnalysisPoint)
772+
if does_namespacing(sys)
773+
return [ap]
774+
else
775+
return [renamespace(sys, ap)]
776+
end
777+
end
772778
canonicalize_ap(sys::AbstractSystem, ap) = [ap]
773779
function canonicalize_ap(sys::AbstractSystem, aps::Vector)
774780
mapreduce(Base.Fix1(canonicalize_ap, sys), vcat, aps; init = [])

src/systems/diffeqs/odesystem.jl

+10-4
Original file line numberDiff line numberDiff line change
@@ -169,7 +169,11 @@ struct ODESystem <: AbstractODESystem
169169
"""
170170
substitutions::Any
171171
"""
172-
If a model `sys` is complete, then `sys.x` no longer performs namespacing.
172+
If false, then `sys.x` no longer performs namespacing.
173+
"""
174+
namespacing::Bool
175+
"""
176+
If true, denotes the model will not be modified any further.
173177
"""
174178
complete::Bool
175179
"""
@@ -207,8 +211,8 @@ struct ODESystem <: AbstractODESystem
207211
connector_type, preface, cevents,
208212
devents, parameter_dependencies, assertions = Dict{BasicSymbolic, String}(),
209213
metadata = nothing, gui_metadata = nothing, is_dde = false,
210-
tstops = [], tearing_state = nothing,
211-
substitutions = nothing, complete = false, index_cache = nothing,
214+
tstops = [], tearing_state = nothing, substitutions = nothing,
215+
namespacing = true, complete = false, index_cache = nothing,
212216
discrete_subsystems = nothing, solved_unknowns = nothing,
213217
split_idxs = nothing, ignored_connections = nothing, parent = nothing;
214218
checks::Union{Bool, Int} = true)
@@ -218,6 +222,7 @@ struct ODESystem <: AbstractODESystem
218222
check_parameters(ps, iv)
219223
check_equations(deqs, iv)
220224
check_equations(equations(cevents), iv)
225+
check_subsystems(systems)
221226
end
222227
if checks == true || (checks & CheckUnits) > 0
223228
u = __get_unit_type(dvs, ps, iv)
@@ -228,7 +233,8 @@ struct ODESystem <: AbstractODESystem
228233
ctrl_jac, Wfact, Wfact_t, name, description, systems, defaults, guesses, torn_matching,
229234
initializesystem, initialization_eqs, schedule, connector_type, preface,
230235
cevents, devents, parameter_dependencies, assertions, metadata,
231-
gui_metadata, is_dde, tstops, tearing_state, substitutions, complete, index_cache,
236+
gui_metadata, is_dde, tstops, tearing_state, substitutions, namespacing,
237+
complete, index_cache,
232238
discrete_subsystems, solved_unknowns, split_idxs, ignored_connections, parent)
233239
end
234240
end

src/systems/diffeqs/sdesystem.jl

+11-8
Original file line numberDiff line numberDiff line change
@@ -139,7 +139,11 @@ struct SDESystem <: AbstractODESystem
139139
"""
140140
gui_metadata::Union{Nothing, GUIMetadata}
141141
"""
142-
If a model `sys` is complete, then `sys.x` no longer performs namespacing.
142+
If false, then `sys.x` no longer performs namespacing.
143+
"""
144+
namespacing::Bool
145+
"""
146+
If true, denotes the model will not be modified any further.
143147
"""
144148
complete::Bool
145149
"""
@@ -166,7 +170,7 @@ struct SDESystem <: AbstractODESystem
166170
guesses, initializesystem, initialization_eqs, connector_type,
167171
cevents, devents, parameter_dependencies, assertions = Dict{
168172
BasicSymbolic, Nothing},
169-
metadata = nothing, gui_metadata = nothing,
173+
metadata = nothing, gui_metadata = nothing, namespacing = true,
170174
complete = false, index_cache = nothing, parent = nothing, is_scalar_noise = false,
171175
is_dde = false,
172176
isscheduled = false;
@@ -184,6 +188,7 @@ struct SDESystem <: AbstractODESystem
184188
if is_scalar_noise && neqs isa AbstractMatrix
185189
throw(ArgumentError("Noise equations ill-formed. Received a matrix of noise equations of size $(size(neqs)), but `is_scalar_noise` was set to `true`. Scalar noise is only compatible with an `AbstractVector` of noise equations."))
186190
end
191+
check_subsystems(systems)
187192
end
188193
if checks == true || (checks & CheckUnits) > 0
189194
u = __get_unit_type(dvs, ps, iv)
@@ -192,8 +197,8 @@ struct SDESystem <: AbstractODESystem
192197
new(tag, deqs, neqs, iv, dvs, ps, tspan, var_to_name, ctrls, observed, tgrad, jac,
193198
ctrl_jac, Wfact, Wfact_t, name, description, systems,
194199
defaults, guesses, initializesystem, initialization_eqs, connector_type, cevents,
195-
devents, parameter_dependencies, assertions, metadata, gui_metadata, complete,
196-
index_cache, parent, is_scalar_noise, is_dde, isscheduled)
200+
devents, parameter_dependencies, assertions, metadata, gui_metadata, namespacing,
201+
complete, index_cache, parent, is_scalar_noise, is_dde, isscheduled)
197202
end
198203
end
199204

@@ -218,7 +223,6 @@ function SDESystem(deqs::AbstractVector{<:Equation}, neqs::AbstractArray, iv, dv
218223
assertions = Dict{BasicSymbolic, String}(),
219224
metadata = nothing,
220225
gui_metadata = nothing,
221-
complete = false,
222226
index_cache = nothing,
223227
parent = nothing,
224228
is_scalar_noise = false,
@@ -274,7 +278,7 @@ function SDESystem(deqs::AbstractVector{<:Equation}, neqs::AbstractArray, iv, dv
274278
ctrl_jac, Wfact, Wfact_t, name, description, systems, defaults, guesses,
275279
initializesystem, initialization_eqs, connector_type,
276280
cont_callbacks, disc_callbacks, parameter_dependencies, assertions, metadata, gui_metadata,
277-
complete, index_cache, parent, is_scalar_noise, is_dde; checks = checks)
281+
true, false, index_cache, parent, is_scalar_noise, is_dde; checks = checks)
278282
end
279283

280284
function SDESystem(sys::ODESystem, neqs; kwargs...)
@@ -321,9 +325,8 @@ function SDESystem(eqs::Vector{Equation}, noiseeqs::AbstractArray, iv; kwargs...
321325
throw(ArgumentError("Variable $dv in noise equations is not an unknown of the system."))
322326
end
323327
algevars = setdiff(allunknowns, diffvars)
324-
325328
return SDESystem(eqs, noiseeqs, iv, Iterators.flatten((diffvars, algevars)),
326-
[ps; collect(noiseps)]; kwargs...)
329+
[collect(ps); collect(noiseps)]; kwargs...)
327330
end
328331

329332
function SDESystem(eq::Equation, noiseeqs::AbstractArray, args...; kwargs...)

src/systems/discrete_system/discrete_system.jl

+9-3
Original file line numberDiff line numberDiff line change
@@ -97,7 +97,11 @@ struct DiscreteSystem <: AbstractDiscreteSystem
9797
"""
9898
substitutions::Any
9999
"""
100-
If a model `sys` is complete, then `sys.x` no longer performs namespacing.
100+
If false, then `sys.x` no longer performs namespacing.
101+
"""
102+
namespacing::Bool
103+
"""
104+
If true, denotes the model will not be modified any further.
101105
"""
102106
complete::Bool
103107
"""
@@ -114,14 +118,15 @@ struct DiscreteSystem <: AbstractDiscreteSystem
114118
observed, name, description, systems, defaults, guesses, initializesystem,
115119
initialization_eqs, preface, connector_type, parameter_dependencies = Equation[],
116120
metadata = nothing, gui_metadata = nothing,
117-
tearing_state = nothing, substitutions = nothing,
121+
tearing_state = nothing, substitutions = nothing, namespacing = true,
118122
complete = false, index_cache = nothing, parent = nothing,
119123
isscheduled = false;
120124
checks::Union{Bool, Int} = true)
121125
if checks == true || (checks & CheckComponents) > 0
122126
check_independent_variables([iv])
123127
check_variables(dvs, iv)
124128
check_parameters(ps, iv)
129+
check_subsystems(systems)
125130
end
126131
if checks == true || (checks & CheckUnits) > 0
127132
u = __get_unit_type(dvs, ps, iv)
@@ -130,7 +135,8 @@ struct DiscreteSystem <: AbstractDiscreteSystem
130135
new(tag, discreteEqs, iv, dvs, ps, tspan, var_to_name, observed, name, description,
131136
systems, defaults, guesses, initializesystem, initialization_eqs,
132137
preface, connector_type, parameter_dependencies, metadata, gui_metadata,
133-
tearing_state, substitutions, complete, index_cache, parent, isscheduled)
138+
tearing_state, substitutions, namespacing, complete, index_cache, parent,
139+
isscheduled)
134140
end
135141
end
136142

src/systems/discrete_system/implicit_discrete_system.jl

+9-3
Original file line numberDiff line numberDiff line change
@@ -96,7 +96,11 @@ struct ImplicitDiscreteSystem <: AbstractDiscreteSystem
9696
"""
9797
substitutions::Any
9898
"""
99-
If a model `sys` is complete, then `sys.x` no longer performs namespacing.
99+
If false, then `sys.x` no longer performs namespacing.
100+
"""
101+
namespacing::Bool
102+
"""
103+
If true, denotes the model will not be modified any further.
100104
"""
101105
complete::Bool
102106
"""
@@ -113,14 +117,15 @@ struct ImplicitDiscreteSystem <: AbstractDiscreteSystem
113117
observed, name, description, systems, defaults, guesses, initializesystem,
114118
initialization_eqs, preface, connector_type, parameter_dependencies = Equation[],
115119
metadata = nothing, gui_metadata = nothing,
116-
tearing_state = nothing, substitutions = nothing,
120+
tearing_state = nothing, substitutions = nothing, namespacing = true,
117121
complete = false, index_cache = nothing, parent = nothing,
118122
isscheduled = false;
119123
checks::Union{Bool, Int} = true)
120124
if checks == true || (checks & CheckComponents) > 0
121125
check_independent_variables([iv])
122126
check_variables(dvs, iv)
123127
check_parameters(ps, iv)
128+
check_subsystems(systems)
124129
end
125130
if checks == true || (checks & CheckUnits) > 0
126131
u = __get_unit_type(dvs, ps, iv)
@@ -129,7 +134,8 @@ struct ImplicitDiscreteSystem <: AbstractDiscreteSystem
129134
new(tag, discreteEqs, iv, dvs, ps, tspan, var_to_name, observed, name, description,
130135
systems, defaults, guesses, initializesystem, initialization_eqs,
131136
preface, connector_type, parameter_dependencies, metadata, gui_metadata,
132-
tearing_state, substitutions, complete, index_cache, parent, isscheduled)
137+
tearing_state, substitutions, namespacing, complete, index_cache, parent,
138+
isscheduled)
133139
end
134140
end
135141

src/systems/jumps/jumpsystem.jl

+8-3
Original file line numberDiff line numberDiff line change
@@ -116,7 +116,11 @@ struct JumpSystem{U <: ArrayPartition} <: AbstractTimeDependentSystem
116116
"""
117117
gui_metadata::Union{Nothing, GUIMetadata}
118118
"""
119-
If a model `sys` is complete, then `sys.x` no longer performs namespacing.
119+
If false, then `sys.x` no longer performs namespacing.
120+
"""
121+
namespacing::Bool
122+
"""
123+
If true, denotes the model will not be modified any further.
120124
"""
121125
complete::Bool
122126
"""
@@ -130,12 +134,13 @@ struct JumpSystem{U <: ArrayPartition} <: AbstractTimeDependentSystem
130134
systems, defaults, guesses, initializesystem, initialization_eqs, connector_type,
131135
cevents, devents,
132136
parameter_dependencies, metadata = nothing, gui_metadata = nothing,
133-
complete = false, index_cache = nothing, isscheduled = false;
137+
namespacing = true, complete = false, index_cache = nothing, isscheduled = false;
134138
checks::Union{Bool, Int} = true) where {U <: ArrayPartition}
135139
if checks == true || (checks & CheckComponents) > 0
136140
check_independent_variables([iv])
137141
check_variables(unknowns, iv)
138142
check_parameters(ps, iv)
143+
check_subsystems(systems)
139144
end
140145
if checks == true || (checks & CheckUnits) > 0
141146
u = __get_unit_type(unknowns, ps, iv)
@@ -145,7 +150,7 @@ struct JumpSystem{U <: ArrayPartition} <: AbstractTimeDependentSystem
145150
observed, name, description, systems, defaults, guesses, initializesystem,
146151
initialization_eqs,
147152
connector_type, cevents, devents, parameter_dependencies, metadata,
148-
gui_metadata, complete, index_cache, isscheduled)
153+
gui_metadata, namespacing, complete, index_cache, isscheduled)
149154
end
150155
end
151156
function JumpSystem(tag, ap, iv, states, ps, var_to_name, args...; kwargs...)

src/systems/nonlinear/nonlinearsystem.jl

+8-3
Original file line numberDiff line numberDiff line change
@@ -95,7 +95,11 @@ struct NonlinearSystem <: AbstractTimeIndependentSystem
9595
"""
9696
substitutions::Any
9797
"""
98-
If a model `sys` is complete, then `sys.x` no longer performs namespacing.
98+
If false, then `sys.x` no longer performs namespacing.
99+
"""
100+
namespacing::Bool
101+
"""
102+
If true, denotes the model will not be modified any further.
99103
"""
100104
complete::Bool
101105
"""
@@ -112,17 +116,18 @@ struct NonlinearSystem <: AbstractTimeIndependentSystem
112116
tag, eqs, unknowns, ps, var_to_name, observed, jac, name, description,
113117
systems, defaults, guesses, initializesystem, initialization_eqs, connector_type,
114118
parameter_dependencies = Equation[], metadata = nothing, gui_metadata = nothing,
115-
tearing_state = nothing, substitutions = nothing,
119+
tearing_state = nothing, substitutions = nothing, namespacing = true,
116120
complete = false, index_cache = nothing, parent = nothing,
117121
isscheduled = false; checks::Union{Bool, Int} = true)
118122
if checks == true || (checks & CheckUnits) > 0
119123
u = __get_unit_type(unknowns, ps)
120124
check_units(u, eqs)
125+
check_subsystems(systems)
121126
end
122127
new(tag, eqs, unknowns, ps, var_to_name, observed, jac, name, description,
123128
systems, defaults, guesses, initializesystem, initialization_eqs,
124129
connector_type, parameter_dependencies, metadata, gui_metadata, tearing_state,
125-
substitutions, complete, index_cache, parent, isscheduled)
130+
substitutions, namespacing, complete, index_cache, parent, isscheduled)
126131
end
127132
end
128133

0 commit comments

Comments
 (0)