Bayesian adjustment of panel test error

devtools::load_all()
#> ℹ Loading testerror
#> ℹ Re-compiling testerror (debug build)
#> ── R CMD INSTALL ───────────────────────────────────────────────────────────────
#> * installing *source* package ‘testerror’ ...
#> ** using staged installation
#> * DONE (testerror)
library(tidyverse)
#> ── Attaching core tidyverse packages ──────────────────────── tidyverse 2.0.0 ──
#> ✔ dplyr     1.1.4     ✔ readr     2.1.5
#> ✔ forcats   1.0.0     ✔ stringr   1.5.1
#> ✔ ggplot2   3.5.1     ✔ tibble    3.2.1
#> ✔ lubridate 1.9.4     ✔ tidyr     1.3.1
#> ✔ purrr     1.0.4     
#> ── Conflicts ────────────────────────────────────────── tidyverse_conflicts() ──
#> ✖ readr::edition_get()   masks testthat::edition_get()
#> ✖ dplyr::filter()        masks stats::filter()
#> ✖ purrr::is_null()       masks testthat::is_null()
#> ✖ dplyr::lag()           masks stats::lag()
#> ✖ readr::local_edition() masks testthat::local_edition()
#> ✖ dplyr::matches()       masks tidyr::matches(), testthat::matches()
#> ℹ Use the conflicted package (<http://conflicted.r-lib.org/>) to force all conflicts to become errors
library(testerror)
options(mc.cores = 2) # parallel::detectCores())
rstan::rstan_options(auto_write = FALSE)
here::i_am("vignettes/testerror.Rmd")
#> here() starts at /tmp/RtmpQCiq8S/Rbuild21786f6f7914/testerror
source(here::here("vignettes/formatting.R"))

BinaxNOW test results


binax_pos = 26
binax_n = 786

# Fit beta distributions to sinclair 2013 data:
# We apply a widening to increase uncertainty in sensitivity
binax_sens = beta_params(median = 0.740, lower = 0.666, upper = 0.823, widen = 5)
binax_spec = beta_params(median = 0.972, lower=  0.927, upper = 0.998)

tmp1 = testerror::bayesian_true_prevalence_model(
  pos_obs = binax_pos,
  n_obs = binax_n,
  sens = binax_sens,
  spec = binax_spec,
  model_type = "logit"
)
#> recompiling to avoid crashing R session
#> Warning: There were 43 divergent transitions after warmup. See
#> https://mc-stan.org/misc/warnings.html#divergent-transitions-after-warmup
#> to find out why this is a problem and how to eliminate them.
#> Warning: Examine the pairs() plot to diagnose sampling problems
#> Warning: Tail Effective Samples Size (ESS) is too low, indicating posterior variances and tail quantiles may be unreliable.
#> Running the chains for more iterations may help. See
#> https://mc-stan.org/misc/warnings.html#tail-ess

binax_summ = tmp1$summary %>%
  dplyr::transmute(
  Test = "Binax",
  Positivity = sprintf("%d/%d (%1.2f%%)", binax_pos, binax_n, binax_pos/binax_n*100),
  `Estimated prevalence` = prevalence.label,
  `Sensitivity` = sens.label,
  `Specificity` = spec.label,
  `Method` = prevalence.method
  )

binax_summ %>% default_table()
TestPositivityEstimated prevalenceSensitivitySpecificityMethod
Binax26/786 (3.31%)0.37% [0.00% — 3.40%]74.95% [53.19% — 89.04%]97.21% [95.71% — 98.89%]bayes (logit)

UAD1 results

For this description we are going to assume we have only count data for panel and components, and we have limited information about component tests.


# data from Forstner et al (2019)
components = tibble::tibble(
  serotype = factor(c("1", "3", "4", "5", "6A", "6B", "7F", "9V", "14", "18C", "19A", "19F", "23F")),
  pos = c(2, 30, 2, 1, 4, 0, 7, 1, 1, 3, 2, 3, 4),
  n = 796
)

uad1_pos = 59
uad1_n = 796

# control group data
# negatives at 
uad1_spec = spec_prior() %>% 
  # data from Pride et al
  update_posterior(0,17)

# Datd from Pride et al
uad1_controls = tibble::tibble(
  serotype = factor(c("1", "3", "4", "5", "6A", "6B", "7F", "9V", "14", "18C", "19A", "19F", "23F")),
  false_neg_diseased = c(0L, 1L, 0L, 0L, 0L, 0L, 0L, 0L, 0L, 0L, 0L, 0L, 0L),
  n_diseased = c(7, 1, 3, 1, 1, 0, 4, 2, 7, 0, 6, 0, 2),
  # In the design of UAD1 the cut off is calibrated on 400 negative samples and 
  # set to make 2 positive
  
  # Forstner et al (2019) control group
  # In the control group of 397 non-CAP patients, 1 patient had to
  # be excluded because of indeterminable results (0.3%, see Fig. 1)
  # and SSUAD was positive in 3 patients (0.8%). Of those 3 non-CAP
  # patients, 2 were positive for serotype 18C and 1 person for two
  # pneumococcal serotypes (5, 7F).

  false_pos_controls = 2+c(0L, 0L, 0L, 1L, 0L, 0L, 1L, 0L, 0L, 2L, 0L, 0L, 0L),
  n_controls = 400+396
)


uad1_sens = sens_prior() %>% 
  # data from Pride et al
  update_posterior(17,17)

With this we can use a bayesian model to construct adjusted estimates


corr3 = testerror::bayesian_panel_true_prevalence_model(
  panel_pos_obs = uad1_pos,
  panel_n_obs = uad1_n,
  panel_name = "PCV13",
  
  pos_obs = components$pos,
  n_obs = components$n,
  test_names = components$serotype,
  
  false_pos_controls = uad1_controls$false_pos_controls,
  n_controls = uad1_controls$n_controls,
  false_neg_diseased = uad1_controls$false_neg_diseased,
  n_diseased = uad1_controls$n_diseased,
  
  panel_sens = uad1_sens,
  model_type = "logit"
)
#> recompiling to avoid crashing R session
corr3$summary %>% 
  mutate(
    pos_obs = c(components$pos, uad1_pos),
    n_obs = c(components$n, uad1_n)
  ) %>%
  dplyr::transmute(
  Test = test,
  Positivity = sprintf("%d/%d (%1.2f%%)", pos_obs, n_obs, pos_obs/n_obs*100),
  `Estimated prevalence` = prevalence.label,
  `Sensitivity` = sens.label,
  `Specificity` = spec.label,
  `Method` = prevalence.method
) %>% 
  bind_rows(binax_summ) %>%
  default_table() %>% huxtable::set_top_border(row = huxtable::final(2), col=huxtable::everywhere, value = 1)
TestPositivityEstimated prevalenceSensitivitySpecificityMethod
12/796 (0.25%)0.03% [0.00% — 0.43%]97.53% [77.74% — 99.93%]99.81% [99.51% — 99.96%]bayes (logit)
330/796 (3.77%)5.64% [2.95% — 22.98%]61.54% [14.74% — 96.64%]99.80% [99.30% — 99.98%]bayes (logit)
42/796 (0.25%)0.02% [0.00% — 0.44%]96.04% [60.88% — 99.91%]99.81% [99.51% — 99.96%]bayes (logit)
51/796 (0.13%)0.01% [0.00% — 0.30%]93.72% [33.60% — 99.86%]99.79% [99.48% — 99.95%]bayes (logit)
6A4/796 (0.50%)0.15% [0.00% — 1.01%]93.92% [32.98% — 99.89%]99.75% [99.38% — 99.96%]bayes (logit)
6B0/796 (0.00%)0.01% [0.00% — 0.32%]87.96% [9.02% — 99.83%]99.90% [99.67% — 99.99%]bayes (logit)
7F7/796 (0.88%)0.35% [0.01% — 1.27%]96.73% [67.01% — 99.91%]99.61% [99.10% — 99.91%]bayes (logit)
9V1/796 (0.13%)0.02% [0.00% — 0.31%]95.10% [48.76% — 99.90%]99.86% [99.58% — 99.97%]bayes (logit)
141/796 (0.13%)0.01% [0.00% — 0.31%]97.59% [78.42% — 99.91%]99.85% [99.57% — 99.97%]bayes (logit)
18C3/796 (0.38%)0.02% [0.00% — 0.65%]88.69% [8.79% — 99.84%]99.62% [99.24% — 99.86%]bayes (logit)
19A2/796 (0.25%)0.02% [0.00% — 0.45%]97.29% [74.87% — 99.93%]99.81% [99.49% — 99.97%]bayes (logit)
19F3/796 (0.38%)0.07% [0.00% — 0.99%]89.48% [9.25% — 99.86%]99.77% [99.43% — 99.96%]bayes (logit)
23F4/796 (0.50%)0.15% [0.00% — 0.86%]95.52% [51.16% — 99.89%]99.75% [99.37% — 99.96%]bayes (logit)
PCV1359/796 (7.41%)6.81% [4.17% — 23.15%]67.12% [22.24% — 94.63%]96.92% [95.88% — 97.83%]bayes (logit)
Binax26/786 (3.31%)0.37% [0.00% — 3.40%]74.95% [53.19% — 89.04%]97.21% [95.71% — 98.89%]bayes (logit)

Sensitivity analysis with different UAD test parameters


# senstivity / specificity from Kakiuchi et al 

kak_uad1_sens = beta_params(median = 0.741, lower = 0.537, upper = 0.889)
kak_uad1_spec = beta_params(median = 0.954, lower = 0.917, upper = 0.978) %>% 
  # data from Pride et al
  update_posterior(17,17)

corr4 = testerror::bayesian_panel_true_prevalence_model(
  panel_pos_obs = uad1_pos,
  panel_n_obs = uad1_n,
  panel_name = "PCV13",
  
  pos_obs = components$pos,
  n_obs = components$n,
  test_names = components$serotype,
  
  false_pos_controls = uad1_controls$false_pos_controls,
  n_controls = uad1_controls$n_controls,
  false_neg_diseased = uad1_controls$false_neg_diseased,
  n_diseased = uad1_controls$n_diseased,
  
  panel_sens = kak_uad1_sens,
  panel_spec = kak_uad1_spec,
  model_type="logit"
)
#> recompiling to avoid crashing R session
#> recompiling to avoid crashing R session
#> Warning: There were 2 divergent transitions after warmup. See
#> https://mc-stan.org/misc/warnings.html#divergent-transitions-after-warmup
#> to find out why this is a problem and how to eliminate them.
#> Warning: Examine the pairs() plot to diagnose sampling problems

corr4$summary %>% 
  mutate(
    pos_obs = c(components$pos, uad1_pos),
    n_obs = c(components$n, uad1_n)
  ) %>%
  dplyr::transmute(
  Test = test,
  Positivity = sprintf("%d/%d (%1.2f%%)", pos_obs, n_obs, pos_obs/n_obs*100),
  `Estimated prevalence` = prevalence.label,
  `Sensitivity` = sens.label,
  `Specificity` = spec.label,
  `Method` = prevalence.method
) %>% bind_rows(
  binax_summ
) %>% 
  default_table() %>% huxtable::set_top_border(row = huxtable::final(2), col=huxtable::everywhere, value = 1)
TestPositivityEstimated prevalenceSensitivitySpecificityMethod
12/796 (0.25%)0.02% [0.00% — 0.44%]97.48% [77.27% — 99.93%]99.80% [99.48% — 99.96%]bayes (logit)
330/796 (3.77%)5.06% [3.18% — 8.38%]66.17% [41.49% — 87.67%]99.78% [99.26% — 99.97%]bayes (logit)
42/796 (0.25%)0.02% [0.00% — 0.43%]96.18% [61.34% — 99.90%]99.80% [99.48% — 99.96%]bayes (logit)
51/796 (0.13%)0.01% [0.00% — 0.32%]93.69% [33.69% — 99.87%]99.78% [99.44% — 99.95%]bayes (logit)
6A4/796 (0.50%)0.15% [0.00% — 0.95%]93.76% [32.48% — 99.87%]99.74% [99.35% — 99.95%]bayes (logit)
6B0/796 (0.00%)0.01% [0.00% — 0.30%]88.06% [7.11% — 99.86%]99.90% [99.65% — 99.99%]bayes (logit)
7F7/796 (0.88%)0.31% [0.00% — 1.24%]96.59% [67.01% — 99.90%]99.56% [99.06% — 99.90%]bayes (logit)
9V1/796 (0.13%)0.02% [0.00% — 0.32%]95.40% [47.96% — 99.89%]99.85% [99.57% — 99.97%]bayes (logit)
141/796 (0.13%)0.01% [0.00% — 0.30%]97.40% [78.04% — 99.93%]99.84% [99.58% — 99.97%]bayes (logit)
18C3/796 (0.38%)0.02% [0.00% — 0.62%]88.01% [8.35% — 99.84%]99.60% [99.19% — 99.85%]bayes (logit)
19A2/796 (0.25%)0.02% [0.00% — 0.44%]97.18% [73.99% — 99.93%]99.79% [99.48% — 99.96%]bayes (logit)
19F3/796 (0.38%)0.06% [0.00% — 0.89%]88.90% [8.03% — 99.83%]99.76% [99.41% — 99.95%]bayes (logit)
23F4/796 (0.50%)0.13% [0.00% — 0.87%]95.57% [49.38% — 99.90%]99.73% [99.35% — 99.95%]bayes (logit)
PCV1359/796 (7.41%)6.04% [4.31% — 9.20%]71.55% [50.76% — 86.02%]96.69% [95.73% — 97.50%]bayes (logit)
Binax26/786 (3.31%)0.37% [0.00% — 3.40%]74.95% [53.19% — 89.04%]97.21% [95.71% — 98.89%]bayes (logit)