Skip to content

Commit

Permalink
add 1 before fitting final week when creating shifted cases (#612)
Browse files Browse the repository at this point in the history
* fit to x+1 and avoid touching zeroes

* add news item

* add details

* add clearer example

* don't use gt for seeding in nonmechanistic model

* add reviewer
  • Loading branch information
sbfnk authored Mar 26, 2024
1 parent 3494e3f commit 2d28525
Show file tree
Hide file tree
Showing 6 changed files with 85 additions and 28 deletions.
1 change: 1 addition & 0 deletions NEWS.md
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@
* Updated the parameterisation of the dispersion term `phi` to be `phi = 1 / sqrt_phi ^ 2` rather than the previous parameterisation `phi = 1 / sqrt(sqrt_phi)` based on the suggested prior [here](https://github.com/stan-dev/stan/wiki/Prior-Choice-Recommendations#story-when-the-generic-prior-fails-the-case-of-the-negative-binomial) and the performance benefits seen in the `epinowcast` package (see [here](https://github.com/epinowcast/epinowcast/blob/8eff560d1fd8305f5fb26c21324b2bfca1f002b4/inst/stan/epinowcast.stan#L314)). By @seabbs in #487 and reviewed by @sbfnk.
* Added an `na` argument to `obs_opts()` that allows the user to specify whether NA values in the data should be interpreted as missing or accumulated in the next non-NA data point. By @sbfnk in #534 and reviewed by @seabbs.
* Growth rates are now calculated directly from the infection trajectory as `log I(t) - log I(t - 1)`. Originally by @seabbs in #213, finished by @sbfnk in #610 and reviewed by @seabbs.
* Fixed a bug when using the nonmechanistic model that could lead to explosive growth. By @sbfnk in #612 and reviewed by @jamesmbaazam.

# EpiNow2 1.4.0

Expand Down
46 changes: 37 additions & 9 deletions R/create.R
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,17 @@ create_complete_cases <- function(cases) {
#' [estimate_infections()] to generate the mean shifted prior on which the back
#' calculation method (see [backcalc_opts()]) is based.
#'
#' @details
#' The function first shifts all the data back in time by `shift` days (thus
#' discarding the first `shift` days of data) and then applies a centred
#' rolling mean of length `smoothing_window` to the shifted data except for
#' the final period. The final period (the forecast horizon plus half the
#' smoothing window) is instead replaced by a log-linear model fit (with 1
#' added to the data for fitting to avoid zeroes and later subtracted again),
#' projected to the end of the forecast horizon. The initial part of the data
#' (corresponding to the length of the smoothing window) is then removed, and
#' any non-integer resulting values rounded up.
#'
#' @param smoothing_window Numeric, the rolling average smoothing window
#' to apply. Must be odd in order to be defined as a centred average.
#'
Expand All @@ -114,7 +125,24 @@ create_complete_cases <- function(cases) {
#' @return A `<data.frame>` for shifted reported cases
#' @export
#' @examples
#' create_shifted_cases(example_confirmed, 7, 14, 7)
#' shift <- 7
#' horizon <- 7
#' smoothing_window <- 14
#' ## add NAs for horizon
#' cases <- create_clean_reported_cases(example_confirmed, horizon = horizon)
#' ## add zeroes initially
#' cases <- data.table::rbindlist(list(
#' data.table::data.table(
#' date = seq(
#' min(cases$date) - smoothing_window,
#' min(cases$date) - 1,
#' by = "days"
#' ),
#' confirm = 0, breakpoint = 0
#' ),
#' cases
#' ))
#' create_shifted_cases(cases, shift, smoothing_window, horizon)
create_shifted_cases <- function(reported_cases, shift,
smoothing_window, horizon) {
shifted_reported_cases <- data.table::copy(reported_cases)[
Expand All @@ -128,18 +156,18 @@ create_shifted_cases <- function(reported_cases, shift,
confirm := runner::mean_run(
confirm, k = smoothing_window, lag = -floor(smoothing_window / 2)
)
][
,
confirm := data.table::fifelse(confirm == 0, 1, confirm) # nolint
]

## Forecast trend on reported cases using the last week of data
final_week <- data.table::data.table(
confirm = shifted_reported_cases[1:(.N - horizon - shift)][
max(1, .N - 6):.N]$confirm)[,
final_period <- data.table::data.table(
confirm =
shifted_reported_cases[!is.na(confirm)][
max(1, .N - smoothing_window):.N
]$confirm
)[,
t := seq_len(.N)
]
lm_model <- stats::lm(log(confirm) ~ t, data = final_week)
lm_model <- stats::lm(log(confirm + 1) ~ t, data = final_period)
## Estimate unreported future infections using a log linear model
shifted_reported_cases <- shifted_reported_cases[
,
Expand All @@ -151,7 +179,7 @@ create_shifted_cases <- function(reported_cases, shift,
,
confirm := data.table::fifelse(
t >= 7,
exp(lm_model$coefficients[1] + lm_model$coefficients[2] * t),
exp(lm_model$coefficients[1] + lm_model$coefficients[2] * t) - 1,
confirm
)
][, t := NULL]
Expand Down
2 changes: 1 addition & 1 deletion R/estimate_infections.R
Original file line number Diff line number Diff line change
Expand Up @@ -164,7 +164,7 @@ estimate_infections <- function(reported_cases,
# Record earliest date with data
start_date <- min(reported_cases$date, na.rm = TRUE)

seeding_time <- get_seeding_time(delays, generation_time)
seeding_time <- get_seeding_time(delays, generation_time, rt)

# Create mean shifted reported cases as prior
reported_cases <- data.table::rbindlist(list(
Expand Down
18 changes: 6 additions & 12 deletions R/get.R
Original file line number Diff line number Diff line change
Expand Up @@ -319,20 +319,14 @@ get_regions_with_most_reports <- function(reported_cases,
##'
##' The seeding time is set to the mean of the specified delays, constrained
##' to be at least the maximum generation time
##' @param delays Delays specified using distribution functions such as
##' [Gamma()] or [LogNormal()]
##' @param generation_time Generation specified using distribution functions
##' such as [Gamma()] or [LogNormal()]
##' @inheritParams estimate_infections
##' @return An integer seeding time
get_seeding_time <- function(delays, generation_time) {
get_seeding_time <- function(delays, generation_time, rt = rt_opts()) {
# Estimate the mean delay -----------------------------------------------
seeding_time <- sum(mean(delays, ignore_uncertainty = TRUE))
if (seeding_time < 1) {
seeding_time <- 1
} else {
seeding_time <- round(seeding_time)
if (!is.null(rt)) {
## make sure we have at least (length of total gt pmf - 1) seeding time
seeding_time <- max(seeding_time, sum(max(generation_time)))
}
## make sure we have at least (length of total gt pmf - 1) seeding time
seeding_time <- max(seeding_time, sum(max(generation_time)))
return(seeding_time)
return(max(round(seeding_time), 1))
}
30 changes: 29 additions & 1 deletion man/create_shifted_cases.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

16 changes: 11 additions & 5 deletions man/get_seeding_time.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

0 comments on commit 2d28525

Please sign in to comment.