Skip to content

epirecipes/sir-julia

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

sir-julia

Various implementations of the classical SIR model in Julia

Try the notebooks out in Binder:

Binder

Model considered

The model equations are as follows.

$$ \begin{align*} \dfrac{\mathrm dS}{\mathrm dt} &= -\frac{\beta c S I}{N}, \\ \dfrac{\mathrm dI}{\mathrm dt} &= \frac{\beta c S I}{N} - \gamma I,\\ \dfrac{\mathrm dR}{\mathrm dt} &= \gamma I, \\ S(t) + I(t) + R(t) &= N \end{align*} $$

Here's a description of the underlying SIR model.

  • The ordinary differential equation model considers:
    • Susceptible, S, with initial condition S(0)=990
    • Infected, I, with initial condition, I(0)=10
    • Recovered, R, with initial condition R(0)=10
    • Total population, N=S+I+R=1000
  • Susceptible individuals make contacts with others at rate c (=10.0), with the probability of a contact with an infectious person being I/N. With probability β (=0.05), an infected person will infect a susceptible given a contact.
  • Infected individuals recover at a per-capita rate Îł (=0.25).

There are two types of parameterization commonly used in this project; the 'standard' version, that considers the number of individuals in the S, I, and R groups, and an alternative version, in which the dynamics of transmission (βSI/N) and recovery (γI) are modelled directly, with S, I, and R being calculated based on these dynamics and the initial conditions for S, I and R.

Simulation with different types of model

The above process can be represented in different kinds of ways:

Ordinary differential equations

Integral equations

Stochastic differential equations

Function maps

Stochastic Markov models

Jump processes

Finite state projection

Petri nets

Stock and flow models

Discrete event simulations

Agent-based models

Other representations

Optimal control

Composing models

Building models from smaller, re-usable components make it easier to build complex models quickly, and also makes it easier to document the development of these models.

Generating simulated data

We usually do not observe the trajectory of susceptible, infected, and recovered individuals. Rather, we often obtain data in terms of new cases aggregated over a particular timescale (e.g. a day or a week).

Use of callbacks

Inference

In addition to the above examples of simulation, there are also examples of inference of the parameters of the model using counts of new cases. Although these are toy examples, they provide the building blocks for more complex situations.

Deterministic models

Stochastic models

Equilibrium analysis

Identifiability

In conducting inference, it is important to know the extent to which parameters are identifiable from the available data.

Uncertainty

Incorporating uncertainty in its many forms is important for using models to make decisions.

Probabilistic integration

When solving continuous-time models like ODEs, the discretization can lead to numerical errors. Probabilistic integration treats this error as a statistical problem to capture the uncertainty in the model outputs generated using the solver.

Global sensitivity

Local sensitivity

Likelihood intervals

Surrogate models

Flexible models

Extensions

Interoperability with other languages

While this repository is mainly about Julia, it is also possible to use Julia to call code written in other languages. Here are some examples of how to define the vector field of an ODE in C, Python, and R.

Using ccall

Many languages can compile to shared libraries that can be accessed via ccall. Here are examples of how to define the vector field of an ODE in various languages, and call it using ccall.

Using Python and R

Comments on implementations

Note that the implementations and choice of parameters may be suboptimal, and are intended to illustrate more-or-less the same underlying biological process with different mathematical representations. Additional optimizations may be obtained e.g. by using StaticArrays.

I've also tried to transform parameterisations in discrete time as closely as possible to their continuous counterparts. Please see the great work by Linda Allen for how these different representations compare.

Types of output

Thanks to Weave.jl, Julia Markdown files (in tutorials/) are converted into multiple formats.

Running notebooks

git clone https://github.com/epirecipes/sir-julia
cd sir-julia

Then launch julia and run the following.

cd(@__DIR__)
import IJulia
IJulia.notebook(;dir="notebook")

Adding new examples

Plans for new examples are typically posted on the Issues page.

To add an example, make a new subdirectory in the tutorials directory, and add a Julia Markdown (.jmd) document to it. Set the beginning to something like the following:

# Agent-based model using Agents.jl
Simon Frost (@sdwfrost), 2020-04-27

Suggested sections:

  • Introduction
  • Libraries
  • Utility functions
  • Transitions
  • Time domain
  • Initial conditions
  • Parameter values
  • Random number seed
  • Running the model
  • Post-processing
  • Plotting
  • Benchmarking

Change to the root directory of the repository and run the following from within Julia; you will need Weave.jl and any dependencies from the tutorial.

include("build.jl")
weave_all() # or e.g. weave_folder("abm") for an individual tutorial

If additional packages are added, then these need to be added to build_project_toml.jl, which when run, will regenerate Project.toml.

Acknowledgements

Examples use the following libraries (see the Project.toml file for a full list of dependencies):

Parts of the code were taken from @ChrisRackauckas DiffEqTutorials, which comes highly recommended.

About

Various implementations of the classical SIR model in Julia

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages