Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Spatial ssa support #663

Closed
wants to merge 138 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
138 commits
Select commit Hold shift + click to select a range
0d32835
Begin tests
TorkelE May 4, 2023
8256e21
Reupload everything
TorkelE May 4, 2023
2355316
add graph imports
TorkelE May 4, 2023
ef14e79
Start preparing tests
TorkelE May 4, 2023
5f34876
test updates
TorkelE May 5, 2023
e64f67f
update tests
TorkelE May 5, 2023
d91ee2b
test fix
TorkelE May 5, 2023
0d58d6c
update
TorkelE Jun 28, 2023
99273d0
test update
TorkelE Jun 29, 2023
63cfb97
formating
TorkelE Jun 29, 2023
7db52c4
change time requirements
TorkelE Jun 29, 2023
9594396
use SIR test
TorkelE Jun 29, 2023
573791c
update
TorkelE Jun 29, 2023
22436ca
update
TorkelE Jun 30, 2023
e42f131
More tests
TorkelE Jun 30, 2023
be32b6c
test improvement
TorkelE Jul 2, 2023
305dcf2
add a test
TorkelE Jul 2, 2023
a08b3b3
update and more tests
TorkelE Jul 2, 2023
9764b2a
format
TorkelE Jul 2, 2023
3ac33cc
internal test update
TorkelE Jul 2, 2023
22e2164
fix
TorkelE Jul 2, 2023
435bbd6
updaye
TorkelE Jul 3, 2023
0b6c577
add test and fixes
TorkelE Jul 3, 2023
6878684
format
TorkelE Jul 3, 2023
40ef2d9
remove old file
TorkelE Jul 4, 2023
c12ed86
Comment out timing tests
TorkelE Jul 6, 2023
62956d6
Enable symbolics in diffusion reactions
TorkelE Jul 6, 2023
10e1e17
format
TorkelE Jul 6, 2023
11e4e5e
small fix
TorkelE Jul 6, 2023
cca2b4e
remove BenchmarkTools test dependency
TorkelE Jul 13, 2023
0df5cee
use Num's everywhere
TorkelE Aug 1, 2023
845d5b6
update
TorkelE Aug 1, 2023
5e9c2dc
use sym properly
TorkelE Aug 2, 2023
e30e768
add test
TorkelE Aug 2, 2023
59f92b4
another update
TorkelE Aug 2, 2023
587d00e
update
TorkelE Aug 2, 2023
aaf3559
Internal revamp and spatial jump support
TorkelE Aug 12, 2023
c672cef
Add spatial tests.
TorkelE Aug 12, 2023
3e50e74
format
TorkelE Aug 12, 2023
74f4042
format
TorkelE Aug 12, 2023
dbded65
test fix
TorkelE Aug 12, 2023
74cf628
Remove Jump Stuff
TorkelE Aug 13, 2023
3bfb270
fix
TorkelE Aug 13, 2023
a502fee
fix
TorkelE Aug 13, 2023
a8b5580
test update
TorkelE Aug 13, 2023
8e768dc
Merge branch 'master' of /~https://github.com/SciML/Catalyst.jl into la…
TorkelE Aug 14, 2023
83808fb
Spatial Jump Implementation
TorkelE Aug 27, 2023
cddc9fb
test update
TorkelE Aug 27, 2023
3bb908e
test update
TorkelE Aug 27, 2023
a5f58bf
Spatial tests file reordering
TorkelE Aug 27, 2023
3ac5d38
testupdate
TorkelE Aug 27, 2023
0891dc1
testupdate
TorkelE Aug 27, 2023
5d9e129
Fix stableRNG usage
TorkelE Sep 2, 2023
cefc59e
Update src/lattice_reaction_system_diffusion.jl
TorkelE Sep 2, 2023
5a3c09c
add ABC test and reset make_majumps
TorkelE Sep 2, 2023
e0d7f3f
remove tests for future functionality
TorkelE Sep 2, 2023
fa91011
explanatory note
TorkelE Sep 3, 2023
fb5977f
Drop formatting changes in `compound.jl`.
Vilin97 Sep 3, 2023
f00289c
use functors
TorkelE Sep 3, 2023
7a60c19
functor update
TorkelE Sep 3, 2023
8fb32c7
Fix test.
Vilin97 Sep 3, 2023
1cdd23f
Construct massaction jumps without building a `JumpProblem`.
Vilin97 Sep 4, 2023
cbb2dbd
Fix comment.
Vilin97 Sep 4, 2023
f910f92
revamp test and start SplitApplyCombine removal
TorkelE Sep 11, 2023
7eafc0d
fix broadcasting issue
TorkelE Sep 11, 2023
66408ec
reorder files.
TorkelE Sep 11, 2023
eb79f66
Move spatial reactions to separate file
TorkelE Sep 11, 2023
11507b5
wip
TorkelE Sep 11, 2023
0b7a426
updates
TorkelE Sep 12, 2023
e5ef053
Finish LatticeReactionSystem revamp
TorkelE Sep 12, 2023
e46e1ad
redo test networks
TorkelE Sep 12, 2023
ee755a2
LatticeReactionSystem revamp
TorkelE Sep 12, 2023
3bc52f7
test updates
TorkelE Sep 13, 2023
f9607c2
More tests
TorkelE Sep 13, 2023
19c9a67
Remove SplitApplyCombine dependency
TorkelE Sep 13, 2023
146d90a
More tests
TorkelE Sep 13, 2023
2219ece
fix
TorkelE Sep 13, 2023
b7831fb
Additional tests
TorkelE Sep 15, 2023
09a8584
Update performance benchmarks (not run, by saved in repo)
TorkelE Sep 15, 2023
a2f8088
Spatial Jump Implementation
TorkelE Aug 27, 2023
1d187db
Spatial tests file reordering
TorkelE Aug 27, 2023
11734ef
testupdate
TorkelE Aug 27, 2023
cc370ea
testupdate
TorkelE Aug 27, 2023
bcee84e
Fix stableRNG usage
TorkelE Sep 2, 2023
4558d0d
Update src/lattice_reaction_system_diffusion.jl
TorkelE Sep 2, 2023
cfe5f88
add ABC test and reset make_majumps
TorkelE Sep 2, 2023
e1cd568
remove tests for future functionality
TorkelE Sep 2, 2023
c6776fc
explanatory note
TorkelE Sep 3, 2023
abc2b7d
Drop formatting changes in `compound.jl`.
Vilin97 Sep 3, 2023
3af9d6f
Fix test.
Vilin97 Sep 3, 2023
7aee979
Construct massaction jumps without building a `JumpProblem`.
Vilin97 Sep 4, 2023
0109d62
Fix comment.
Vilin97 Sep 4, 2023
dc983bb
Make spatial SSA part up to date.
TorkelE Sep 15, 2023
ab75e6f
Merge remote-tracking branch 'origin/spatial_SSA_support' into spatia…
TorkelE Sep 15, 2023
5e6191a
update
TorkelE Sep 16, 2023
dde4ee9
update
TorkelE Sep 16, 2023
27daaf9
up
TorkelE Sep 16, 2023
c43e2c1
test testing
TorkelE Sep 16, 2023
7cccf7b
test test
TorkelE Sep 16, 2023
bbcb06d
test update
TorkelE Sep 16, 2023
ea24729
test testing
TorkelE Sep 16, 2023
0d4da1a
fix
TorkelE Sep 16, 2023
83e1f27
test update
TorkelE Sep 16, 2023
89965e2
Split of utility function and handle un-directed graph input properly
TorkelE Sep 17, 2023
619589f
Test fix
TorkelE Sep 17, 2023
6550e97
fix
TorkelE Sep 17, 2023
03f7f20
update history file.
TorkelE Sep 17, 2023
60794af
Spatial Jump Implementation
TorkelE Aug 27, 2023
bc6952f
Spatial tests file reordering
TorkelE Aug 27, 2023
2c4efc0
testupdate
TorkelE Aug 27, 2023
1758ae3
testupdate
TorkelE Aug 27, 2023
7cbbcf7
Fix stableRNG usage
TorkelE Sep 2, 2023
724136d
add ABC test and reset make_majumps
TorkelE Sep 2, 2023
3ec45e5
remove tests for future functionality
TorkelE Sep 2, 2023
57a5028
Drop formatting changes in `compound.jl`.
Vilin97 Sep 3, 2023
dae6e04
Fix test.
Vilin97 Sep 3, 2023
9928608
Spatial Jump Implementation
TorkelE Aug 27, 2023
9da5f31
Fix stableRNG usage
TorkelE Sep 2, 2023
fb52503
Update src/lattice_reaction_system_diffusion.jl
TorkelE Sep 2, 2023
2ade53f
add ABC test and reset make_majumps
TorkelE Sep 2, 2023
9a0d403
explanatory note
TorkelE Sep 3, 2023
381262f
Drop formatting changes in `compound.jl`.
Vilin97 Sep 3, 2023
024fd0d
Construct massaction jumps without building a `JumpProblem`.
Vilin97 Sep 4, 2023
0f865ed
Fix comment.
Vilin97 Sep 4, 2023
b2a5732
Make spatial SSA part up to date.
TorkelE Sep 15, 2023
abd7bf4
update
TorkelE Sep 16, 2023
00d917e
update
TorkelE Sep 16, 2023
2831fd3
up
TorkelE Sep 16, 2023
197cd23
test testing
TorkelE Sep 16, 2023
5f01557
test test
TorkelE Sep 16, 2023
76fd25c
test update
TorkelE Sep 16, 2023
2454f48
test testing
TorkelE Sep 16, 2023
89c1e0e
fix
TorkelE Sep 16, 2023
87f7ccc
test update
TorkelE Sep 16, 2023
fa50613
Merge remote-tracking branch 'origin/spatial_SSA_support' into spatia…
TorkelE Sep 20, 2023
b282002
mereg
TorkelE Sep 20, 2023
8f5b10f
fix merge
TorkelE Sep 20, 2023
92cdc84
update history file
TorkelE Sep 20, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 13 additions & 0 deletions HISTORY.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,19 @@
# Breaking updates and feature summaries across releases

## Catalyst unreleased (master branch)
- Simulation of spatial ODEs now supported. For full details, please see /~https://github.com/SciML/Catalyst.jl/pull/644 and upcoming documentation.
- Simulations of spatial Jumps are now supported. For full details see: /~https://github.com/SciML/Catalyst.jl/pull/663. For documentation on the spatial SSA solvers used, pelase see: https://docs.sciml.ai/JumpProcesses/stable/tutorials/spatial/.
- LatticeReactionSystem structure represents a spatiral reaction network:
```julia
rn = @reaction_network begin
(p,d), 0 <--> X
end
tr = @transport_reaction D X
lattice = Graphs.grid([5, 5])
lrs = LatticeReactionSystem(rn, [tr], lattice)
```



## Catalyst 13.4
- Added the ability to create species that represent chemical compounds and know
Expand Down
22 changes: 21 additions & 1 deletion src/Catalyst.jl
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,8 @@ import ModelingToolkit: check_variables,
get_unit, check_equations

import Base: (==), hash, size, getindex, setindex, isless, Sort.defalg, length, show
import MacroTools, Graphs
import MacroTools
import Graphs, Graphs.DiGraph, Graphs.SimpleGraph, Graphs.SimpleDiGraph, Graphs.vertices, Graphs.edges, Graphs.add_vertices!, Graphs.nv, Graphs.ne
import DataStructures: OrderedDict, OrderedSet
import Parameters: @with_kw_noshow

Expand Down Expand Up @@ -101,4 +102,23 @@ export @compound
export components, iscompound, coefficients
export balance_reaction

### Spatial Reaction Networks ###

# spatial reactions
include("spatial_reaction_systems/spatial_reactions.jl")
export TransportReaction, transport_reactions, @transport_reaction
export isedgeparameter

# lattice reaction systems
include("spatial_reaction_systems/lattice_reaction_systems.jl")
export LatticeReactionSystem
export spatial_species, vertex_parameters, edge_parameters

# variosu utility functions
include("spatial_reaction_systems/utility.jl")

# spatial lattice ode and jump systems.
include("spatial_reaction_systems/spatial_ODE_systems.jl")
include("spatial_reaction_systems/spatial_Jump_systems.jl")

end # module
66 changes: 66 additions & 0 deletions src/spatial_reaction_systems/lattice_reaction_systems.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
### Lattice Reaction Network Structure ###
# Desribes a spatial reaction network over a graph.
struct LatticeReactionSystem{S,T} # <: MT.AbstractTimeDependentSystem # Adding this part messes up show, disabling me from creating LRSs
"""The reaction system within each comaprtment."""
rs::ReactionSystem{S}
"""The spatial reactions defined between individual nodes."""
spatial_reactions::Vector{T}
"""The graph on which the lattice is defined."""
lattice::SimpleDiGraph{Int64}

# Derrived values.
"""The number of compartments."""
nV::Int64
"""The number of edges."""
nE::Int64
"""The number of species."""
nS::Int64
"""Whenever the initial input was a digraph."""
init_digraph::Bool
"""Species that may move spatially."""
spat_species::Vector{BasicSymbolic{Real}}
"""Parameters which values are tied to edges (adjacencies).."""
edge_parameters::Vector{BasicSymbolic{Real}}

function LatticeReactionSystem(rs::ReactionSystem{S},
spatial_reactions::Vector{T},
lattice::DiGraph; init_digraph = true) where {S, T}
(T <: AbstractSpatialReaction) || error("The second argument must be a vector of AbstractSpatialReaction subtypes.") # There probably some better way to acertain that T has that type. Not sure how.
spat_species = unique(vcat(spatial_species.(spatial_reactions)...))
rs_edge_parameters = filter(isedgeparameter, parameters(rs))
srs_edge_parameters = setdiff(vcat(parameters.(spatial_reactions)...), parameters(rs))
edge_parameters = unique([rs_edge_parameters; srs_edge_parameters])

foreach(sr -> check_spatial_reaction_validity(rs, sr; edge_parameters=edge_parameters), spatial_reactions)
return new{S,T}(rs, spatial_reactions, lattice, nv(lattice), ne(lattice), length(unique([species(rs); spat_species])), init_digraph, spat_species, edge_parameters)
end
end
function LatticeReactionSystem(rs, srs, lat::SimpleGraph)
return LatticeReactionSystem(rs, srs, DiGraph(lat); init_digraph = false)
end
function LatticeReactionSystem(rs, sr::AbstractSpatialReaction, lat)
return LatticeReactionSystem(rs, [sr], lat)
end
function LatticeReactionSystem(rs, sr::AbstractSpatialReaction, lat::SimpleGraph)
return LatticeReactionSystem(rs, [sr], DiGraph(lat); init_digraph = false)
end

### Lattice ReactionSystem Getters ###

# Get all species.
species(lrs::LatticeReactionSystem) = unique([species(lrs.rs); lrs.spat_species])
# Get all species that may be transported.
spatial_species(lrs::LatticeReactionSystem) = lrs.spat_species

# Get all parameters.
ModelingToolkit.parameters(lrs::LatticeReactionSystem) = unique([parameters(lrs.rs); lrs.edge_parameters])
# Get all parameters which values are tied to vertexes (compartments).
vertex_parameters(lrs::LatticeReactionSystem) = setdiff(parameters(lrs), edge_parameters(lrs))
# Get all parameters which values are tied to edges (adjacencies).
edge_parameters(lrs::LatticeReactionSystem) = lrs.edge_parameters

# Gets the lrs name (same as rs name).
ModelingToolkit.nameof(lrs::LatticeReactionSystem) = nameof(lrs.rs)

# Checks if a lattice reaction system is a pure (linear) transport reaction system.
is_transport_system(lrs::LatticeReactionSystem) = all(typeof.(lrs.spatial_reactions) .== TransportReaction)
54 changes: 54 additions & 0 deletions src/spatial_reaction_systems/spatial_Jump_systems.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
### JumpProblem ###

# Builds a spatial DiscreteProblem from a Lattice Reaction System.

# Creates a DiscreteProblem from a LatticeReactionSystem.
function DiffEqBase.DiscreteProblem(lrs::LatticeReactionSystem, u0_in, tspan, p_in = DiffEqBase.NullParameters(), args...; kwargs...)
is_transport_system(lrs) || error("Currently lattice Jump simulations only supported when all spatial reactions are transport reactions.")

# Converts potential symmaps to varmaps.
u0_in = symmap_to_varmap(lrs, u0_in)
p_in = (p_in isa Tuple{<:Any,<:Any}) ? (symmap_to_varmap(lrs, p_in[1]),symmap_to_varmap(lrs, p_in[2])) : symmap_to_varmap(lrs, p_in)

# Converts u0 and p to Vector{Vector{Float64}} form.
u0 = lattice_process_u0(u0_in, species(lrs), lrs.nV)
pC, pD = lattice_process_p(p_in, vertex_parameters(lrs), edge_parameters(lrs), lrs)
# Creates DiscreteProblem.
return DiscreteProblem(lrs.rs, u0, tspan, (pC, pD), args...; kwargs...)
end

# Builds a spatial JumpProblem from a DiscreteProblem containg a Lattice Reaction System.
function JumpProcesses.JumpProblem(lrs::LatticeReactionSystem, dprob, aggregator, args...; name = nameof(lrs.rs), combinatoric_ratelaws = get_combinatoric_ratelaws(lrs.rs),checks = false, kwargs...)
# Error checks.
dprob.p isa Tuple{Vector{Vector{Float64}}, Vector{Vector{Float64}}} || dprob.p isa Vector{Vector} || error("Parameters in input DiscreteProblem is of an unexpected type: $(typeof(dprob.p)). Was a LatticeReactionProblem passed into the DiscreteProblem when it was created?") # The second check (Vector{Vector} is needed becaus on the CI server somehow the Tuple{..., ...} is covnerted into a Vector[..., ...]). It does not happen when I run tests locally, so no ideal how to fix.
any(length.(dprob.p[1]) .> 1) && error("Spatial reaction rates are currently not supported in lattice jump simulations.")

# Creates JumpProblem.
hopping_constants = make_hopping_constants(dprob, lrs)
___dprob = DiscreteProblem(reshape(dprob.u0, lrs.nS, lrs.nV), dprob.tspan, first.(dprob.p[1]))
majumps_ = make_majumps(___dprob, lrs.rs)
return JumpProblem(___dprob, aggregator, majumps_, hopping_constants = hopping_constants, spatial_system = lrs.lattice, save_positions = (true, false))
end

# Creates the hopping constants from a discrete problem and a lattice reaction system.
function make_hopping_constants(dprob::DiscreteProblem, lrs::LatticeReactionSystem)
spatial_rates_dict = Dict(compute_all_spatial_rates(dprob.p[1], dprob.p[2], lrs))
all_diff_rates = [haskey(spatial_rates_dict, s) ? spatial_rates_dict[s] : [0.0] for s in species(lrs)]
hopping_constants = [Vector{Float64}(undef, length(lrs.lattice.fadjlist[j])) for i in 1:(lrs.nS), j in 1:(lrs.nV)]
for (e_idx, e) in enumerate(edges(lrs.lattice)), s_idx in 1:(lrs.nS)
dst_idx = findfirst(isequal(e.dst), lrs.lattice.fadjlist[e.src])
hopping_constants[s_idx, e.src][dst_idx] = get_component_value(all_diff_rates[s_idx], e_idx)
end
return hopping_constants
end

# Creates the mass action jumps from a discrete problem and a reaction system.
function make_majumps(non_spat_dprob, rs::ReactionSystem)
js = convert(JumpSystem, rs)
statetoid = Dict(ModelingToolkit.value(state) => i for (i, state) in enumerate(states(rs)))
eqs = equations(js)
invttype = non_spat_dprob.tspan[1] === nothing ? Float64 : typeof(1 / non_spat_dprob.tspan[2])
p = (non_spat_dprob.p isa DiffEqBase.NullParameters || non_spat_dprob.p === nothing) ? Num[] : non_spat_dprob.p
majpmapper = ModelingToolkit.JumpSysMajParamMapper(js, p; jseqs = eqs, rateconsttype = invttype)
return ModelingToolkit.assemble_maj(eqs.x[1], statetoid, majpmapper)
end
211 changes: 211 additions & 0 deletions src/spatial_reaction_systems/spatial_ODE_systems.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,211 @@
### Spatial ODE Functor Structures ###

# Functor structure containg the information for the forcing function of a spatial ODE with spatial movement on a lattice.
struct LatticeDiffusionODEf{R,S,T}
ofunc::R
nV::Int64
nS::Int64
pV::Vector{Vector{Float64}}
work_pV::Vector{Float64}
enumerated_pV_idx_types::Base.Iterators.Enumerate{BitVector}
spatial_rates::Vector{S}
leaving_rates::Matrix{Float64}
enumerated_edges::T

function LatticeDiffusionODEf(ofunc::R, pV, spatial_rates::Vector{S}, lrs::LatticeReactionSystem) where {R, S, T}
leaving_rates = zeros(length(spatial_rates), lrs.nV)
for (s_idx, rates) in enumerate(last.(spatial_rates)),
(e_idx, e) in enumerate(edges(lrs.lattice))

leaving_rates[s_idx, e.src] += get_component_value(rates, e_idx)
end
work_pV = zeros(lrs.nV)
enumerated_pV_idx_types = enumerate(length.(pV) .== 1)
enumerated_edges = deepcopy(enumerate(edges(lrs.lattice)))
new{R,S,typeof(enumerated_edges)}(ofunc, lrs.nV, lrs.nS, pV, work_pV, enumerated_pV_idx_types, spatial_rates, leaving_rates, enumerated_edges)
end
end

# Functor structure containg the information for the forcing function of a spatial ODE with spatial movement on a lattice.
struct LatticeDiffusionODEjac{S,T}
ofunc::S
nV::Int64
nS::Int64
pV::Vector{Vector{Float64}}
work_pV::Vector{Float64}
enumerated_pV_idx_types::Base.Iterators.Enumerate{BitVector}
sparse::Bool
jac_values::T

function LatticeDiffusionODEjac(ofunc::S, pV, lrs::LatticeReactionSystem, jac_prototype::Union{Nothing, SparseMatrixCSC{Float64, Int64}}, sparse::Bool) where {S, T}
work_pV = zeros(lrs.nV)
enumerated_pV_idx_types = enumerate(length.(pV) .== 1)
jac_values = sparse ? jac_prototype.nzval : Matrix(jac_prototype)
new{S,typeof(jac_values)}(ofunc, lrs.nV, lrs.nS, pV, work_pV, enumerated_pV_idx_types, sparse, jac_values)
end
end

### ODEProblem ###

# Creates an ODEProblem from a LatticeReactionSystem.
function DiffEqBase.ODEProblem(lrs::LatticeReactionSystem, u0_in, tspan,
p_in = DiffEqBase.NullParameters(), args...;
jac = true, sparse = jac, kwargs...)
is_transport_system(lrs) || error("Currently lattice ODE simulations only supported when all spatial reactions are transport reactions.")

# Converts potential symmaps to varmaps.
u0_in = symmap_to_varmap(lrs, u0_in)
p_in = (p_in isa Tuple{<:Any,<:Any}) ? (symmap_to_varmap(lrs, p_in[1]),symmap_to_varmap(lrs, p_in[2])) : symmap_to_varmap(lrs, p_in)

# Converts u0 and p to Vector{Vector{Float64}} form.
u0 = lattice_process_u0(u0_in, species(lrs), lrs.nV)
pV, pE = lattice_process_p(p_in, vertex_parameters(lrs), edge_parameters(lrs), lrs)

# Creates ODEProblem.
ofun = build_odefunction(lrs, pV, pE, jac, sparse)
return ODEProblem(ofun, u0, tspan, pV, args...; kwargs...)
end

# Builds an ODEFunction for a spatial ODEProblem.
function build_odefunction(lrs::LatticeReactionSystem, pV::Vector{Vector{Float64}},
pE::Vector{Vector{Float64}}, use_jac::Bool, sparse::Bool)
# Prepeares (non-spatial) ODE functions and list of spatially moving species and their rates.
ofunc = ODEFunction(convert(ODESystem, lrs.rs); jac = use_jac, sparse = false)
ofunc_sparse = ODEFunction(convert(ODESystem, lrs.rs); jac = use_jac, sparse = true)
spatial_rates_speciesmap = compute_all_spatial_rates(pV, pE, lrs)
spatial_rates = [findfirst(isequal(spat_rates[1]), species(lrs)) => spat_rates[2]
for spat_rates in spatial_rates_speciesmap]

f = LatticeDiffusionODEf(ofunc, pV, spatial_rates, lrs)
jac_prototype = (use_jac || sparse) ?
build_jac_prototype(ofunc_sparse.jac_prototype, spatial_rates,
lrs; set_nonzero = use_jac) : nothing
jac = use_jac ? LatticeDiffusionODEjac(ofunc, pV, lrs, jac_prototype, sparse) : nothing
return ODEFunction(f; jac = jac, jac_prototype = (sparse ? jac_prototype : nothing))
end

# Builds a jacobian prototype. If requested, populate it with the Jacobian's (constant) values as well.
function build_jac_prototype(ns_jac_prototype::SparseMatrixCSC{Float64, Int64}, spatial_rates, lrs::LatticeReactionSystem;
set_nonzero = false)
spat_species = first.(spatial_rates)

# Gets list of indexes for species that move spatially, but are invovled in no other reaction.
only_spat = [(s in spat_species) && !Base.isstored(ns_jac_prototype, s, s)
for s in 1:(lrs.nS)]

# Declares sparse array content.
J_colptr = fill(1, lrs.nV * lrs.nS + 1)
J_nzval = fill(0.0,
lrs.nV * (nnz(ns_jac_prototype) + count(only_spat)) +
length(edges(lrs.lattice)) * length(spatial_rates))
J_rowval = fill(0, length(J_nzval))

# Finds filled elements.
for comp in 1:(lrs.nV), s in 1:(lrs.nS)
col_idx = get_index(comp, s, lrs.nS)

# Column values.
local_elements = in(s, spat_species) *
(length(lrs.lattice.fadjlist[comp]) + only_spat[s])
spatial_elements = -(ns_jac_prototype.colptr[(s + 1):-1:s]...)
J_colptr[col_idx + 1] = J_colptr[col_idx] + local_elements + spatial_elements

# Row values.
rows = ns_jac_prototype.rowval[ns_jac_prototype.colptr[s]:(ns_jac_prototype.colptr[s + 1] - 1)] .+
(comp - 1) * lrs.nS
if in(s, spat_species)
# Finds the location of the spatial_elements, and inserts the elements from the non-spatial part into this.
spatial_rows = (lrs.lattice.fadjlist[comp] .- 1) .* lrs.nS .+ s
split_idx = isempty(rows) ? 1 : findfirst(spatial_rows .> rows[1])
isnothing(split_idx) && (split_idx = length(spatial_rows) + 1)
rows = vcat(spatial_rows[1:(split_idx - 1)], rows,
spatial_rows[split_idx:end])
if only_spat[s]
split_idx = findfirst(rows .> get_index(comp, s, lrs.nS))
isnothing(split_idx) && (split_idx = length(rows) + 1)
insert!(rows, split_idx, get_index(comp, s, lrs.nS))
end
end
J_rowval[J_colptr[col_idx]:(J_colptr[col_idx + 1] - 1)] = rows
end

# Set element values.
if !set_nonzero
J_nzval .= 1.0
else
for (s_idx, (s, rates)) in enumerate(spatial_rates),
(e_idx, edge) in enumerate(edges(lrs.lattice))

col_start = J_colptr[get_index(edge.src, s, lrs.nS)]
col_end = J_colptr[get_index(edge.src, s, lrs.nS) + 1] - 1
column_view = @view J_rowval[col_start:col_end]

# Updates the source value.
val_idx_src = col_start +
findfirst(column_view .== get_index(edge.src, s, lrs.nS)) - 1
J_nzval[val_idx_src] -= get_component_value(rates, e_idx)

# Updates the destination value.
# println()
# println()
# println(col_start)
# println(column_view)
# println(get_index(edge.dst, s, lrs.nS))
# println(col_start)
# println(col_end)
# println(column_view)
val_idx_dst = col_start +
findfirst(column_view .== get_index(edge.dst, s, lrs.nS)) - 1
J_nzval[val_idx_dst] += get_component_value(rates, e_idx)
end
end

return SparseMatrixCSC(lrs.nS * lrs.nV, lrs.nS * lrs.nV, J_colptr, J_rowval, J_nzval)
end

# Defines the forcing functors effect on the (spatial) ODE system.
function (f_func::LatticeDiffusionODEf)(du, u, p, t)
# Updates for non-spatial reactions.
for comp_i::Int64 in 1:(f_func.nV)
f_func.ofunc((@view du[get_indexes(comp_i, f_func.nS)]),
(@view u[get_indexes(comp_i, f_func.nS)]),
view_pV_vector!(f_func.work_pV, p, comp_i, f_func.enumerated_pV_idx_types), t)
end

# Updates for spatial reactions.
for (s_idx, (s, rates)) in enumerate(f_func.spatial_rates)
for comp_i::Int64 in 1:(f_func.nV)
du[get_index(comp_i, s, f_func.nS)] -= f_func.leaving_rates[s_idx, comp_i] *
u[get_index(comp_i, s,
f_func.nS)]
end
for (e_idx::Int64, edge::Graphs.SimpleGraphs.SimpleEdge{Int64}) in f_func.enumerated_edges
du[get_index(edge.dst, s, f_func.nS)] += get_component_value(rates, e_idx) *
u[get_index(edge.src, s,
f_func.nS)]
end
end
end

# Defines the jacobian functors effect on the (spatial) ODE system.
function (jac_func::LatticeDiffusionODEjac)(J, u, p, t)
# Because of weird stuff where the Jacobian is not reset that I don't understand properly.
reset_J_vals!(J)

# Updates for non-spatial reactions.
for comp_i::Int64 in 1:(jac_func.nV)
jac_func.ofunc.jac((@view J[get_indexes(comp_i, jac_func.nS),
get_indexes(comp_i, jac_func.nS)]),
(@view u[get_indexes(comp_i, jac_func.nS)]),
view_pV_vector!(jac_func.work_pV, p, comp_i, jac_func.enumerated_pV_idx_types), t)
end

# Updates for the spatial reactions.
add_spat_J_vals!(J, jac_func)
end
# Resets the jacobian matrix within a jac call.
reset_J_vals!(J::Matrix) = (J .= 0.0)
reset_J_vals!(J::SparseMatrixCSC) = (J.nzval .= 0.0)
# Updates the jacobian matrix with the difussion values.
add_spat_J_vals!(J::SparseMatrixCSC, jac_func::LatticeDiffusionODEjac) = (J.nzval .+= jac_func.jac_values)
add_spat_J_vals!(J::Matrix, jac_func::LatticeDiffusionODEjac) = (J .+= jac_func.jac_values)
Loading