diff --git a/.JuliaFormatter.toml b/.JuliaFormatter.toml index 580b7511..959ad88a 100644 --- a/.JuliaFormatter.toml +++ b/.JuliaFormatter.toml @@ -1 +1,3 @@ style = "sciml" +format_markdown = true +format_docstrings = true \ No newline at end of file diff --git a/README.md b/README.md index 7ccd781b..94ea1103 100644 --- a/README.md +++ b/README.md @@ -16,15 +16,14 @@ Pkg.add("SymbolicIndexingInterface") The symbolic indexing interface has 2 levels: -1. The user level. At the user level, a modeler or engineer simply uses terms from a - domain-specific language (DSL) inside of SciML functionality and will receive the requested - values. For example, if a DSL defines a symbol `x`, then `sol[x]` returns the solution - value(s) for `x`. -2. The DSL system structure level. This is the structure which defines the symbolic indexing - for a given problem/solution. DSLs can tag a constructed problem/solution with this - object in order to endow the SciML tools with the ability to index symbolically according - to the definitions the DSL writer wants. - + 1. The user level. At the user level, a modeler or engineer simply uses terms from a + domain-specific language (DSL) inside of SciML functionality and will receive the requested + values. For example, if a DSL defines a symbol `x`, then `sol[x]` returns the solution + value(s) for `x`. + 2. The DSL system structure level. This is the structure which defines the symbolic indexing + for a given problem/solution. DSLs can tag a constructed problem/solution with this + object in order to endow the SciML tools with the ability to index symbolically according + to the definitions the DSL writer wants. ## Example diff --git a/docs/src/complete_sii.md b/docs/src/complete_sii.md index 047d2162..f026a38b 100644 --- a/docs/src/complete_sii.md +++ b/docs/src/complete_sii.md @@ -5,12 +5,11 @@ This tutorial will show how to define the entire Symbolic Indexing Interface on ```julia struct ExampleSystem - state_index::Dict{Symbol,Int} - parameter_index::Dict{Symbol,Int} - independent_variable::Union{Symbol,Nothing} - defaults::Dict{Symbol, Float64} - # mapping from observed variable to Expr to calculate its value - observed::Dict{Symbol,Expr} + state_index::Dict{Symbol, Int} + parameter_index::Dict{Symbol, Int} + independent_variable::Union{Symbol, Nothing} + # mapping from observed variable to Expr to calculate its value + observed::Dict{Symbol, Expr} end ``` @@ -25,62 +24,62 @@ These are the simple functions which describe how to turn symbols into indices. ```julia function SymbolicIndexingInterface.is_variable(sys::ExampleSystem, sym) - haskey(sys.state_index, sym) + haskey(sys.state_index, sym) end function SymbolicIndexingInterface.variable_index(sys::ExampleSystem, sym) - get(sys.state_index, sym, nothing) + get(sys.state_index, sym, nothing) end function SymbolicIndexingInterface.variable_symbols(sys::ExampleSystem) - collect(keys(sys.state_index)) + collect(keys(sys.state_index)) end function SymbolicIndexingInterface.is_parameter(sys::ExampleSystem, sym) - haskey(sys.parameter_index, sym) + haskey(sys.parameter_index, sym) end function SymbolicIndexingInterface.parameter_index(sys::ExampleSystem, sym) - get(sys.parameter_index, sym, nothing) + get(sys.parameter_index, sym, nothing) end function SymbolicIndexingInterface.parameter_symbols(sys::ExampleSystem) - collect(keys(sys.parameter_index)) + collect(keys(sys.parameter_index)) end function SymbolicIndexingInterface.is_independent_variable(sys::ExampleSystem, sym) - # note we have to check separately for `nothing`, otherwise - # `is_independent_variable(p, nothing)` would return `true`. - sys.independent_variable !== nothing && sym === sys.independent_variable + # note we have to check separately for `nothing`, otherwise + # `is_independent_variable(p, nothing)` would return `true`. + sys.independent_variable !== nothing && sym === sys.independent_variable end function SymbolicIndexingInterface.independent_variable_symbols(sys::ExampleSystem) - sys.independent_variable === nothing ? [] : [sys.independent_variable] + sys.independent_variable === nothing ? [] : [sys.independent_variable] end function SymbolicIndexingInterface.is_time_dependent(sys::ExampleSystem) - sys.independent_variable !== nothing + sys.independent_variable !== nothing end SymbolicIndexingInterface.constant_structure(::ExampleSystem) = true function SymbolicIndexingInterface.all_solvable_symbols(sys::ExampleSystem) - return vcat( - collect(keys(sys.state_index)), - collect(keys(sys.observed)), - ) + return vcat( + collect(keys(sys.state_index)), + collect(keys(sys.observed)) + ) end function SymbolicIndexingInterface.all_symbols(sys::ExampleSystem) - return vcat( - all_solvable_symbols(sys), - collect(keys(sys.parameter_index)), - sys.independent_variable === nothing ? Symbol[] : sys.independent_variable - ) + return vcat( + all_solvable_symbols(sys), + collect(keys(sys.parameter_index)), + sys.independent_variable === nothing ? Symbol[] : sys.independent_variable + ) end function SymbolicIndexingInterface.default_values(sys::ExampleSystem) - return sys.defaults + return sys.defaults end ``` @@ -95,36 +94,38 @@ RuntimeGeneratedFunctions.init(@__MODULE__) # this type accepts `Expr` for observed expressions involving state/parameter/observed # variables -SymbolicIndexingInterface.is_observed(sys::ExampleSystem, sym) = sym isa Expr || sym isa Symbol && haskey(sys.observed, sym) +function SymbolicIndexingInterface.is_observed(sys::ExampleSystem, sym) + sym isa Expr || sym isa Symbol && haskey(sys.observed, sym) +end function SymbolicIndexingInterface.observed(sys::ExampleSystem, sym::Expr) - # generate a function with the appropriate signature - if is_time_dependent(sys) - fn_expr = :( - function gen(u, p, t) - # assign a variable for each state symbol it's value in u - $([:($var = u[$idx]) for (var, idx) in pairs(sys.state_index)]...) - # assign a variable for each parameter symbol it's value in p - $([:($var = p[$idx]) for (var, idx) in pairs(sys.parameter_index)]...) - # assign a variable for the independent variable - $(sys.independent_variable) = t - # return the value of the expression - return $sym - end - ) - else - fn_expr = :( - function gen(u, p) - # assign a variable for each state symbol it's value in u - $([:($var = u[$idx]) for (var, idx) in pairs(sys.state_index)]...) - # assign a variable for each parameter symbol it's value in p - $([:($var = p[$idx]) for (var, idx) in pairs(sys.parameter_index)]...) - # return the value of the expression - return $sym - end - ) - end - return @RuntimeGeneratedFunction(fn_expr) + # generate a function with the appropriate signature + if is_time_dependent(sys) + fn_expr = :( + function gen(u, p, t) + # assign a variable for each state symbol it's value in u + $([:($var = u[$idx]) for (var, idx) in pairs(sys.state_index)]...) + # assign a variable for each parameter symbol it's value in p + $([:($var = p[$idx]) for (var, idx) in pairs(sys.parameter_index)]...) + # assign a variable for the independent variable + $(sys.independent_variable) = t + # return the value of the expression + return $sym + end + ) + else + fn_expr = :( + function gen(u, p) + # assign a variable for each state symbol it's value in u + $([:($var = u[$idx]) for (var, idx) in pairs(sys.state_index)]...) + # assign a variable for each parameter symbol it's value in p + $([:($var = p[$idx]) for (var, idx) in pairs(sys.parameter_index)]...) + # return the value of the expression + return $sym + end + ) + end + return @RuntimeGeneratedFunction(fn_expr) end ``` @@ -136,16 +137,17 @@ defined to always return `false`, and `observed` does not need to be implemented Note that the method definitions are all assuming `constant_structure(p) == true`. In case `constant_structure(p) == false`, the following methods would change: -- `constant_structure(::ExampleSystem) = false` -- `variable_index(sys::ExampleSystem, sym)` would become - `variable_index(sys::ExampleSystem, sym i)` where `i` is the time index at which - the index of `sym` is required. -- `variable_symbols(sys::ExampleSystem)` would become - `variable_symbols(sys::ExampleSystem, i)` where `i` is the time index at which - the variable symbols are required. -- `observed(sys::ExampleSystem, sym)` would become - `observed(sys::ExampleSystem, sym, i)` where `i` is either the time index at which - the index of `sym` is required or a `Vector` of state symbols at the current time index. + + - `constant_structure(::ExampleSystem) = false` + - `variable_index(sys::ExampleSystem, sym)` would become + `variable_index(sys::ExampleSystem, sym i)` where `i` is the time index at which + the index of `sym` is required. + - `variable_symbols(sys::ExampleSystem)` would become + `variable_symbols(sys::ExampleSystem, i)` where `i` is the time index at which + the variable symbols are required. + - `observed(sys::ExampleSystem, sym)` would become + `observed(sys::ExampleSystem, sym, i)` where `i` is either the time index at which + the index of `sym` is required or a `Vector` of state symbols at the current time index. ## Optional methods @@ -163,7 +165,7 @@ them is not necessary. ```julia function SymbolicIndexingInterface.parameter_values(sys::ExampleSystem) - sys.p + sys.p end ``` @@ -179,10 +181,10 @@ Consider the following `ExampleIntegrator` ```julia mutable struct ExampleIntegrator - u::Vector{Float64} - p::Vector{Float64} - t::Float64 - sys::ExampleSystem + u::Vector{Float64} + p::Vector{Float64} + t::Float64 + sys::ExampleSystem end # define a fallback for the interface methods @@ -193,6 +195,7 @@ SymbolicIndexingInterface.current_time(sys::ExampleIntegrator) = sys.t ``` Then the following example would work: + ```julia sys = ExampleSystem(Dict(:x => 1, :y => 2, :z => 3), Dict(:a => 1, :b => 2), :t, Dict()) integrator = ExampleIntegrator([1.0, 2.0, 3.0], [4.0, 5.0], 6.0, sys) @@ -215,10 +218,10 @@ the [`Timeseries`](@ref) trait. The type would then return a timeseries from ```julia struct ExampleSolution - u::Vector{Vector{Float64}} - t::Vector{Float64} - p::Vector{Float64} - sys::ExampleSystem + u::Vector{Vector{Float64}} + t::Vector{Float64} + p::Vector{Float64} + sys::ExampleSystem end # define a fallback for the interface methods @@ -233,6 +236,7 @@ SymbolicIndexingInterface.current_time(sol::ExampleSolution) = sol.t ``` Then the following example would work: + ```julia # using the same system that the ExampleIntegrator used sol = ExampleSolution([[1.0, 2.0, 3.0], [1.5, 2.5, 3.5]], [4.0, 5.0], [6.0, 7.0], sys) @@ -262,8 +266,8 @@ follows: ```julia function SymbolicIndexingInterface.set_state!(integrator::ExampleIntegrator, val, idx) - integrator.u[idx] = val - integrator.u_modified = true + integrator.u[idx] = val + integrator.u_modified = true end ``` @@ -279,24 +283,25 @@ performed for a bulk parameter update. # The `ParameterIndexingProxy` [`ParameterIndexingProxy`](@ref) is a wrapper around another type which implements the -interface and allows using [`getp`](@ref) and [`setp`](@ref) to get and set parameter +interface and allows using [`getp`](@ref) and [`setp`](@ref) to get and set parameter values. This allows for a cleaner interface for parameter indexing. Consider the following example for `ExampleIntegrator`: ```julia function Base.getproperty(obj::ExampleIntegrator, sym::Symbol) - if sym === :ps - return ParameterIndexingProxy(obj) - else - return getfield(obj, sym) - end + if sym === :ps + return ParameterIndexingProxy(obj) + else + return getfield(obj, sym) + end end ``` This enables the following API: ```julia -integrator = ExampleIntegrator([1.0, 2.0, 3.0], [4.0, 5.0], 6.0, Dict(:x => 1, :y => 2, :z => 3), Dict(:a => 1, :b => 2), :t) +integrator = ExampleIntegrator([1.0, 2.0, 3.0], [4.0, 5.0], 6.0, + Dict(:x => 1, :y => 2, :z => 3), Dict(:a => 1, :b => 2), :t) integrator.ps[:a] # 4.0 getp(integrator, :a)(integrator) # functionally the same as above @@ -310,25 +315,25 @@ setp(integrator, :b)(integrator, 3.0) # functionally the same as above The `SymbolicTypeTrait` is used to identify values that can act as symbolic variables. It has three variants: -- [`NotSymbolic`](@ref) for quantities that are not symbolic. This is the default for all - types. -- [`ScalarSymbolic`](@ref) for quantities that are symbolic, and represent a single - logical value. -- [`ArraySymbolic`](@ref) for quantities that are symbolic, and represent an array of - values. Types implementing this trait must return an array of `ScalarSymbolic` variables - of the appropriate size and dimensions when `collect`ed. + - [`NotSymbolic`](@ref) for quantities that are not symbolic. This is the default for all + types. + - [`ScalarSymbolic`](@ref) for quantities that are symbolic, and represent a single + logical value. + - [`ArraySymbolic`](@ref) for quantities that are symbolic, and represent an array of + values. Types implementing this trait must return an array of `ScalarSymbolic` variables + of the appropriate size and dimensions when `collect`ed. The trait is implemented through the [`symbolic_type`](@ref) function. Consider the following example types: ```julia struct MySym - name::Symbol + name::Symbol end struct MySymArr{N} - name::Symbol - size::NTuple{N,Int} + name::Symbol + size::NTuple{N, Int} end ``` @@ -343,10 +348,8 @@ SymbolicIndexingInterface.symbolic_type(::Type{<:MySymArr}) = ArraySymbolic() SymbolicIndexingInterface.hasname(::MySymArr) = true SymbolicIndexingInterface.getname(sym::MySymArr) = sym.name function Base.collect(sym::MySymArr) - [ - MySym(Symbol(sym.name, :_, join(idxs, "_"))) - for idxs in Iterators.product(Base.OneTo.(sym.size)...) - ] + [MySym(Symbol(sym.name, :_, join(idxs, "_"))) + for idxs in Iterators.product(Base.OneTo.(sym.size)...)] end ``` @@ -357,8 +360,8 @@ Introducing a type to represent expression trees: ```julia struct MyExpr - op::Function - args::Vector{Union{MyExpr, MySym, MySymArr, Number, Array}} + op::Function + args::Vector{Union{MyExpr, MySym, MySymArr, Number, Array}} end ``` @@ -366,17 +369,17 @@ end ```julia function symbolic_evaluate(expr::Union{MySym, MySymArr}, syms::Dict) - get(syms, expr, expr) + get(syms, expr, expr) end function symbolic_evaluate(expr::MyExpr, syms::Dict) - for i in eachindex(expr.args) - if expr.args[i] isa Union{MySym, MySymArr, MyExpr} - expr.args[i] = symbolic_evaluate(expr.args[i], syms) + for i in eachindex(expr.args) + if expr.args[i] isa Union{MySym, MySymArr, MyExpr} + expr.args[i] = symbolic_evaluate(expr.args[i], syms) + end + end + if all(x -> symbolic_type(x) === NotSymbolic(), expr.args) + return expr.op(expr.args...) end - end - if all(x -> symbolic_type(x) === NotSymbolic(), expr.args) - return expr.op(expr.args...) - end end ``` @@ -426,7 +429,6 @@ end SymbolicIndexingInterface.parameter_timeseries(fs::ExampleSolution2) = fs.pt # Mark the object as a `Timeseries` object SymbolicIndexingInterface.is_timeseries(::Type{ExampleSolution2}) = Timeseries() - ``` Now we can create an example object and observe the new functionality. Note that diff --git a/docs/src/index.md b/docs/src/index.md index 898f5926..67a9c418 100644 --- a/docs/src/index.md +++ b/docs/src/index.md @@ -16,25 +16,27 @@ Pkg.add("SymbolicIndexingInterface") The symbolic indexing interface has 2 levels: -1. The user level. At the user level, a modeler or engineer simply uses terms from a - domain-specific language (DSL) inside of SciML functionality and will receive the requested - values. For example, if a DSL defines a symbol `x`, then `sol[x]` returns the solution - value(s) for `x`. -2. The DSL system structure level. This is the structure which defines the symbolic indexing - for a given problem/solution. DSLs can tag a constructed problem/solution with this - object in order to endow the SciML tools with the ability to index symbolically according - to the definitions the DSL writer wants. + 1. The user level. At the user level, a modeler or engineer simply uses terms from a + domain-specific language (DSL) inside of SciML functionality and will receive the requested + values. For example, if a DSL defines a symbol `x`, then `sol[x]` returns the solution + value(s) for `x`. + 2. The DSL system structure level. This is the structure which defines the symbolic indexing + for a given problem/solution. DSLs can tag a constructed problem/solution with this + object in order to endow the SciML tools with the ability to index symbolically according + to the definitions the DSL writer wants. ## Contributing -- Please refer to the - [SciML ColPrac: Contributor's Guide on Collaborative Practices for Community Packages](https://github.com/SciML/ColPrac/blob/master/README.md) - for guidance on PRs, issues, and other matters relating to contributing to SciML. -- There are a few community forums: - - the #diffeq-bridged channel in the [Julia Slack](https://julialang.org/slack/) - - [JuliaDiffEq](https://gitter.im/JuliaDiffEq/Lobby) on Gitter - - on the [Julia Discourse forums](https://discourse.julialang.org) - - see also [SciML Community page](https://sciml.ai/community/) + - Please refer to the + [SciML ColPrac: Contributor's Guide on Collaborative Practices for Community Packages](https://github.com/SciML/ColPrac/blob/master/README.md) + for guidance on PRs, issues, and other matters relating to contributing to SciML. + + - There are a few community forums: + + + the #diffeq-bridged channel in the [Julia Slack](https://julialang.org/slack/) + + [JuliaDiffEq](https://gitter.im/JuliaDiffEq/Lobby) on Gitter + + on the [Julia Discourse forums](https://discourse.julialang.org) + + see also [SciML Community page](https://sciml.ai/community/) ## Reproducibility diff --git a/docs/src/simple_sii_sys.md b/docs/src/simple_sii_sys.md index 1a031216..4c2499ec 100644 --- a/docs/src/simple_sii_sys.md +++ b/docs/src/simple_sii_sys.md @@ -56,7 +56,7 @@ sol[:y₁] ``` ```@example symbolcache -sol(1e3, idxs=:y₁) +sol(1e3, idxs = :y₁) ``` However, we did not give names to the parameters or the independent variables. They can diff --git a/docs/src/solution_wrappers.md b/docs/src/solution_wrappers.md index c6599a58..c487a8e7 100644 --- a/docs/src/solution_wrappers.md +++ b/docs/src/solution_wrappers.md @@ -5,9 +5,9 @@ All its methods can simply be forwarded to that object. To do so, SymbolicIndexi provides the [`symbolic_container`](@ref) method. For example, ```julia -struct MySolutionWrapper{T<:SciMLBase.AbstractTimeseriesSolution} - sol::T - # other properties... +struct MySolutionWrapper{T <: SciMLBase.AbstractTimeseriesSolution} + sol::T + # other properties... end symbolic_container(sys::MySolutionWrapper) = sys.sol diff --git a/docs/src/usage.md b/docs/src/usage.md index 693256d4..04349d23 100644 --- a/docs/src/usage.md +++ b/docs/src/usage.md @@ -12,6 +12,7 @@ We recommend any DSL implementing the symbolic indexing interface to link to thi as a full description of the functionality. !!! note + While this tutorial focuses on demonstrating the symbolic indexing interface for ODEs, note that the same functionality works across all of the other problem types, such as optimization problems, nonlinear problems, nonlinear solutions, etc. @@ -76,19 +77,19 @@ sol[(t, w)] ``` ```@example Usage -sol(1.3, idxs=x) +sol(1.3, idxs = x) ``` ```@example Usage -sol(1.3, idxs=[x, w]) +sol(1.3, idxs = [x, w]) ``` ```@example Usage -sol(1.3, idxs=[:y, :z]) +sol(1.3, idxs = [:y, :z]) ``` ```@example Usage -plot(sol, idxs=x) +plot(sol, idxs = x) ``` If necessary, `Symbol`s can be used to refer to variables. This is only valid for @@ -117,6 +118,7 @@ sol[solvedvariables] # equivalent to sol[variable_symbols(sol)] This does not include the observed variable `w`. To include observed variables in the output, the following shorthand is used: + ```@example Usage sol[allvariables] # equivalent to sol[all_variable_symbols(sol)] ``` @@ -150,6 +152,7 @@ expression, and thus it needs to be provided to the `ProblemState`. Parameters cannot be obtained using this syntax, and instead require using [`getp`](@ref) and [`setp`](@ref). !!! note + The reason why parameters use a separate syntax is to be able to ensure type stability of the `sol[x]` indexing. Without separating the parameter indexing, the return type of symbolic indexing could be anything a parameter can be, which is general is not the same @@ -201,6 +204,7 @@ parameter_values(prob) ``` !!! note + These getters and setters generate high-performance functions for the specific chosen symbols or collection of symbols. Caching the getter/setter function and reusing it on other problem/solution instances can be the key to achieving good performance. Note @@ -214,5 +218,5 @@ changing the type of values in the buffer (for example, changing the value of a from `Float64` to `Float32`). ```@example Usage -remake_buffer(sys, prob.p, Dict(σ => 1f0, ρ => 2f0, β => 3f0)) +remake_buffer(sys, prob.p, Dict(σ => 1.0f0, ρ => 2.0f0, β => 3.0f0)) ``` diff --git a/src/parameter_indexing.jl b/src/parameter_indexing.jl index d7b73cf7..9c5c3525 100644 --- a/src/parameter_indexing.jl +++ b/src/parameter_indexing.jl @@ -31,7 +31,7 @@ By default, this function returns `parameter_values(p)` regardless of `i`, and o to be specialized for timeseries objects where parameter values are not constant at all times. The resultant object should be indexable using [`parameter_values`](@ref). -If this function is implemented, [`parameter_values_at_state_time`](@ref) must be +If this function is implemented, [`parameter_values_at_state_time`](@ref) must be implemented for [`getu`](@ref) to work correctly. """ function parameter_values_at_time end diff --git a/src/state_indexing.jl b/src/state_indexing.jl index 782fc7bb..e2bb11d6 100644 --- a/src/state_indexing.jl +++ b/src/state_indexing.jl @@ -41,7 +41,6 @@ the state value is saved. In this case, the two-argument version of the function also be implemented to efficiently return the time at timestep `i`. By default, the two- argument method calls `current_time(p)[i]` - See: [`is_timeseries`](@ref) """ function current_time end diff --git a/src/trait.jl b/src/trait.jl index ea7964a9..a1999105 100644 --- a/src/trait.jl +++ b/src/trait.jl @@ -77,7 +77,7 @@ not present as keys in `syms`. The function can take additional keyword arguments to control implementation-specific behavior. -This is already implemented for +This is already implemented for `symbolic_evaluate(expr::Union{Symbol, Expr}, syms::Dict)`. """ function symbolic_evaluate(expr::Union{Symbol, Expr}, syms::Dict)