measles epiworld logo

ForeSITE Group CRAN status R-CMD-check CRANlogs downloads License: MIT codecov status

Overview

The measles package is a specialized spinoff from epiworldR, focusing exclusively on measles epidemiological models. This package provides fast, agent-based models (ABMs) for studying measles transmission dynamics, vaccination strategies, and intervention policies.

Built on the powerful epiworld C++ library, these models leverage the speed and flexibility of epiworldR while providing specialized functionality for measles outbreak modeling.

You can learn more about the ongoing efforts to model measles outbreaks in ForeSITE’s measles repository: https://github.com/EpiForeSITE/measles.

Features

Models Included

The package includes three measles-specific models:

  1. ModelMeaslesSchool: A SEIHR (Susceptible-Exposed-Infectious-Hospitalized-Recovered) model designed for school settings with isolation and quarantine policies.

  2. ModelMeaslesMixing: A measles model with population mixing between different groups, including vaccination, quarantine, isolation, and contact tracing mechanisms.

  3. ModelMeaslesMixingRiskQuarantine: An advanced mixing model with risk-based quarantine strategies that assign different quarantine durations based on exposure risk levels (high, medium, low).

Installation

You can install the measles package from GitHub:

# install.packages("devtools")
devtools::install_github("UofUEpiBio/measles")

Or from R-universe (recommended for the latest development version):

install.packages(
  'measles',
  repos = c(
    'https://uofuepibio.r-universe.dev',
    'https://cloud.r-project.org'
  )
)

Quick Example

Here’s a simple example using the ModelMeaslesSchool:

library(measles)
#> Loading required package: epiworldR
#> Thank you for using epiworldR! Please consider citing it in your work.
#> You can find the citation information by running
#>   citation("epiworldR")

# Create a measles model for a school with 500 students
model_school <- ModelMeaslesSchool(
  n = 500,
  contact_rate = 10 / .9 / 4, # R0 of 10
  prevalence = 1,
  prop_vaccinated = 0.70,
  quarantine_period = -1,
  vax_efficacy = 0.97
)

# Run the simulation
run_multiple(
  model_school,
  ndays = 200,
  seed = 1912,
  nsims = 400,
  saver = make_saver("outbreak_size"),
  nthreads = 4L
)
#> Starting multiple runs (400) using 4 thread(s)
#> _________________________________________________________________________
#> _________________________________________________________________________
#> ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||| done.

We can take a quick look at the last run using the summary() function:

# View results
summary(model_school)
#> ________________________________________________________________________________
#> ________________________________________________________________________________
#> SIMULATION STUDY
#>
#> Name of the model   : (none)
#> Population size     : 500
#> Agents' data        : (none)
#> Number of entities  : 0
#> Days (duration)     : 200 (of 200)
#> Number of viruses   : 1
#> Last run elapsed t  : 1.00ms
#> Total elapsed t     : 115.00ms (400 runs)
#> Last run speed      : 77.94 million agents x day / second
#> Average run speed   : 347.30 million agents x day / second
#> Rewiring            : off
#> Last seed used      : 1264933217
#>
#> Global events:
#>  - Update model (runs daily)
#>
#> Virus(es):
#>  - Measles
#>
#> Tool(s):
#>  - MMR 0.970000
#>
#> Model parameters:
#>  - (IGNORED) Vax improved recovery : 0.0e+00
#>  - Contact rate                    : 2.7778
#>  - Days undetected                 : 2.0000
#>  - Hospitalization period          : 7.0000
#>  - Hospitalization rate            : 0.2000
#>  - Incubation period               : 12.0000
#>  - Isolation period                : 4.0000
#>  - Prodromal period                : 4.0000
#>  - Quarantine period               : -1.0000
#>  - Quarantine willingness          : 1.0000
#>  - Rash period                     : 3.0000
#>  - Transmission rate               : 0.9000
#>  - Vaccination rate                : 0.7000
#>  - Vax efficacy                    : 0.9700
#>
#> Distribution of the population at time 200:
#>   - ( 0) Susceptible             : 499 -> 346
#>   - ( 1) Exposed                 :   1 -> 0
#>   - ( 2) Prodromal               :   0 -> 0
#>   - ( 3) Rash                    :   0 -> 0
#>   - ( 4) Isolated                :   0 -> 0
#>   - ( 5) Isolated Recovered      :   0 -> 0
#>   - ( 6) Detected Hospitalized   :   0 -> 0
#>   - ( 7) Quarantined Exposed     :   0 -> 0
#>   - ( 8) Quarantined Susceptible :   0 -> 0
#>   - ( 9) Quarantined Prodromal   :   0 -> 0
#>   - (10) Quarantined Recovered   :   0 -> 0
#>   - (11) Hospitalized            :   0 -> 0
#>   - (12) Recovered               :   0 -> 154
#>
#> Transition Probabilities:
#>  - Susceptible              1.00  0.00     -     -     -     -     -     -     -     -     -     -     -
#>  - Exposed                     -  0.92  0.08     -     -     -     -     -     -     -     -     -     -
#>  - Prodromal                   -     -  0.77  0.23     -     -     -     -     -     -     -     -     -
#>  - Rash                        -     -     -  0.29  0.18  0.15  0.08     -     -     -     -  0.13  0.16
#>  - Isolated                    -     -     -  0.12  0.35  0.20  0.22     -     -     -     -  0.03  0.08
#>  - Isolated Recovered          -     -     -     -     -  0.56     -     -     -     -     -     -  0.44
#>  - Detected Hospitalized       -     -     -     -     -     -  0.87     -     -     -     -     -  0.13
#>  - Quarantined Exposed         -     -     -     -     -     -     -     -     -     -     -     -     -
#>  - Quarantined Susceptible     -     -     -     -     -     -     -     -     -     -     -     -     -
#>  - Quarantined Prodromal       -     -     -     -     -     -     -     -     -     -     -     -     -
#>  - Quarantined Recovered       -     -     -     -     -     -     -     -     -     -     -     -     -
#>  - Hospitalized                -     -     -     -     -     -     -     -     -     -     -  0.78  0.22
#>  - Recovered                   -     -     -     -     -     -     -     -     -     -     -     -  1.00

And we can extract the results (outbreak size in this case), using the function run_multiple_get_results():

ans <- run_multiple_get_results(model_school)[[1]]
ans <- subset(ans, date == max(date))$outbreak_size
table(ans) |>
  prop.table() |>
  knitr::kable(
    col.names = c("Outbreak Size", "Proportion of Simulations"),
    digits = 3,
    caption = "Distribution of outbreak sizes across 400 simulations"
  )
Outbreak Size Proportion of Simulations
1 0.175
2 0.038
3 0.013
4 0.002
6 0.002
136 0.002
140 0.013
141 0.002
142 0.005
143 0.002
144 0.010
145 0.013
146 0.022
147 0.020
148 0.022
149 0.028
150 0.048
151 0.058
152 0.052
153 0.068
154 0.072
155 0.052
156 0.072
157 0.048
158 0.048
159 0.028
160 0.032
161 0.015
162 0.010
163 0.020
164 0.005
165 0.002

Distribution of outbreak sizes across 400 simulations

Example with Population Mixing

The mixing models allow you to simulate measles spread across different population groups:

# Define three populations
e1 <- entity("Population 1", 3000, as_proportion = FALSE)
e2 <- entity("Population 2", 3000, as_proportion = FALSE)
e3 <- entity("Population 3", 3000, as_proportion = FALSE)

# Define contact matrix (row-stochastic: rows sum to 1)
contact_matrix <- matrix(c(
  0.9, 0.05, 0.05,
  0.1, 0.8,  0.1,
  0.1, 0.2,  0.7
), byrow = TRUE, nrow = 3)

# Create the model
measles_model <- ModelMeaslesMixing(
  n = 9000,
  prevalence = 1 / 9000,
  contact_rate = 15,
  transmission_rate = 0.9,
  vax_efficacy = 0.97,
  prop_vaccinated = 0.95,
  contact_matrix = contact_matrix,
  quarantine_period = 14,
  isolation_period = 10
)

# Add entities and run
measles_model |>
  add_entity(e1) |>
  add_entity(e2) |>
  add_entity(e3)

run_multiple(
  measles_model, ndays = 100, seed = 331,
  nsims = 400,
  saver = make_saver("outbreak_size"),
  nthreads = 4L
)
#> Starting multiple runs (400) using 4 thread(s)
#> _________________________________________________________________________
#> _________________________________________________________________________
#> ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||| done.

summary(measles_model)
#> ________________________________________________________________________________
#> ________________________________________________________________________________
#> SIMULATION STUDY
#>
#> Name of the model   : Measles with Mixing and Quarantine
#> Population size     : 9000
#> Agents' data        : (none)
#> Number of entities  : 3
#> Days (duration)     : 100 (of 100)
#> Number of viruses   : 1
#> Last run elapsed t  : 0.00s
#> Total elapsed t     : 3.00s (400 runs)
#> Last run speed      : 19.84 million agents x day / second
#> Average run speed   : 91.46 million agents x day / second
#> Rewiring            : off
#> Last seed used      : 1428497254
#>
#> Global events:
#>  - Update infected individuals (runs daily)
#>
#> Virus(es):
#>  - Measles
#>
#> Tool(s):
#>  - MMR 0.970000
#>
#> Model parameters:
#>  - (IGNORED) Vax improved recovery : 0.5000
#>  - Contact rate                    : 15.0000
#>  - Contact tracing days prior      : 4.0000
#>  - Contact tracing success rate    : 1.0000
#>  - Days undetected                 : 2.0000
#>  - Hospitalization period          : 7.0000
#>  - Hospitalization rate            : 0.2000
#>  - Incubation period               : 12.0000
#>  - Isolation period                : 10.0000
#>  - Isolation willingness           : 1.0000
#>  - Prodromal period                : 4.0000
#>  - Quarantine period               : 14.0000
#>  - Quarantine willingness          : 1.0000
#>  - Rash period                     : 3.0000
#>  - Transmission rate               : 0.9000
#>  - Vaccination rate                : 0.9500
#>  - Vax efficacy                    : 0.9700
#>
#> Distribution of the population at time 100:
#>   - ( 0) Susceptible             : 8999 -> 8326
#>   - ( 1) Exposed                 :    1 -> 50
#>   - ( 2) Prodromal               :    0 -> 20
#>   - ( 3) Rash                    :    0 -> 6
#>   - ( 4) Isolated                :    0 -> 2
#>   - ( 5) Isolated Recovered      :    0 -> 25
#>   - ( 6) Detected Hospitalized   :    0 -> 16
#>   - ( 7) Quarantined Exposed     :    0 -> 6
#>   - ( 8) Quarantined Susceptible :    0 -> 2
#>   - ( 9) Quarantined Prodromal   :    0 -> 3
#>   - (10) Quarantined Recovered   :    0 -> 0
#>   - (11) Hospitalized            :    0 -> 12
#>   - (12) Recovered               :    0 -> 532
#>
#> Transition Probabilities:
#>  - Susceptible              1.00  0.00     -     -     -     -     -     -  0.00     -     -     -     -
#>  - Exposed                     -  0.89  0.08     -     -     -     -  0.02     -  0.00     -     -     -
#>  - Prodromal                   -     -  0.74  0.25  0.00     -     -     -     -  0.01     -     -     -
#>  - Rash                        -     -     -  0.22  0.22  0.15  0.08     -     -     -     -  0.12  0.19
#>  - Isolated                    -     -     -     -  0.47  0.36  0.18     -     -     -     -     -     -
#>  - Isolated Recovered          -     -     -     -     -  0.88     -     -     -     -     -     -  0.12
#>  - Detected Hospitalized       -     -     -     -     -     -  0.86     -     -     -     -     -  0.14
#>  - Quarantined Exposed         -  0.04  0.00     -     -     -     -  0.89     -  0.07     -     -     -
#>  - Quarantined Susceptible  0.07     -     -     -     -     -     -     -  0.93     -     -     -     -
#>  - Quarantined Prodromal       -     -  0.01     -  0.27     -     -     -     -  0.72     -     -     -
#>  - Quarantined Recovered       -     -     -     -     -     -     -     -     -     -     -     -     -
#>  - Hospitalized                -     -     -     -     -     -     -     -     -     -     -  0.86  0.14
#>  - Recovered                   -     -     -     -     -     -     -     -     -     -     -     -  1.00
ans <- run_multiple_get_results(measles_model)[[1]]
ans <- subset(ans, date == max(date))$outbreak_size |>
  hist(
    main = "Distribution of outbreak sizes across 400 simulations",
    xlab = "Outbreak Size",
    ylab = "Frequency",
    breaks = 20
  )

Relationship to epiworldR

This package is a spinoff from epiworldR, which provides a comprehensive framework for agent-based epidemiological models. While epiworldR includes many different disease models (SIR, SEIR, SIS, etc.), the measles package focuses specifically on measles-related models with specialized features for:

For general-purpose epidemiological modeling or other disease types, please see the epiworldR package.

Citation

If you use the measles package in your research, please cite both this package and epiworldR:

citation("measles")
citation("epiworldR")

Contributing

Contributions are welcome! Please see the measles development guidelines for information on how to contribute.

Acknowledgments

This project was made possible by cooperative agreement CDC-RFA-FT-23-0069 from the CDC’s Center for Forecasting and Outbreak Analytics. Its contents are solely the responsibility of the authors and do not necessarily represent the official views of the Centers for Disease Control and Prevention.