Introduction

q-SIP method workflow

Quantitative stable isotope probing (q-SIP) attempts to deal with the compositional problems of HTS-SIP data by transforming relative abundances with qPCR counts of total gene copies.

More details can be found at:

Hungate, Bruce A., Rebecca L. Mau, Egbert Schwartz, J. Gregory Caporaso, Paul Dijkstra, Natasja van Gestel, Benjamin J. Koch, et al. 2015. “Quantitative Microbial Ecology Through Stable Isotope Probing.” Applied and Environmental Microbiology, August, AEM.02280–15.

HTSSIP can both simulate q-SIP datasets and run the q-SIP analysis (with parallel computation of bootstrap confidence intervals).

Dataset

First, let's load some packages including HTSSIP.

library(dplyr)
library(ggplot2)
library(HTSSIP)

OK. We're going to be using 2 data files:

We'll be using the dataset that we simulated in the HTSSIP_sim vignette.

The phyloseq object is similar to the dataset in the other vignettes.

# HTS-SIP data
physeq_rep3
## phyloseq-class experiment-level object
## otu_table()   OTU Table:         [ 6 taxa and 144 samples ]
## sample_data() Sample Data:       [ 144 samples by 5 sample variables ]

The associated qPCR data is, in this case a list of length = 2.

# qPCR data (list object)
physeq_rep3_qPCR %>% names
## [1] "raw"     "summary"

Really, only the summary table is needed for the next step. You could just use the summary data.frame object and that would be fine. However, you should have just 1 qPCR value per sample. So, if you have 'technical' qPCR replicates (eg., triplicate qPCR reactions per DNA sample), then you need to summarize those replicates to get one value (eg., the mean of all replicate values).

physeq_rep3_qPCR$raw %>% head(n=3)
##   Buoyant_density IS_CONTROL qPCR_tech_rep1 qPCR_tech_rep2 qPCR_tech_rep3
## 1        1.668185       TRUE      374460160      326925692      191209173
## 2        1.680254       TRUE      418974851      846274648     1057133579
## 3        1.679431       TRUE      456070748      431437845      307250348
##                    Sample     Gradient Fraction Treatment Replicate
## 1 12C-Con_rep1_1.668185_1 12C-Con_rep1        1   12C-Con         1
## 2 12C-Con_rep1_1.680254_2 12C-Con_rep1        2   12C-Con         1
## 3 12C-Con_rep1_1.679431_3 12C-Con_rep1        3   12C-Con         1
physeq_rep3_qPCR$summary %>% head(n=3)
##   IS_CONTROL                  Sample Buoyant_density qPCR_tech_rep_mean
## 1      FALSE 13C-Glu_rep1_1.671712_2        1.671712           46598172
## 2      FALSE 13C-Glu_rep1_1.671722_1        1.671722           42299449
## 3      FALSE 13C-Glu_rep1_1.680311_3        1.680311           53536519
##   qPCR_tech_rep_sd     Gradient Fraction Treatment Replicate
## 1          9055833 13C-Glu_rep1        2   13C-Glu         1
## 2         16369298 13C-Glu_rep1        1   13C-Glu         1
## 3         18265667 13C-Glu_rep1        3   13C-Glu         1

q-SIP

Abundance count transformation

First, we'll transform the OTU counts. The following function will do the following:

phyloseq::otu_table(physeq_rep3) %>% .[1:5, 1:5]
## OTU Table:          [5 taxa and 5 samples]
##                      taxa are rows
##       12C-Con_rep1_1.668185_1 12C-Con_rep1_1.680254_2
## OTU.1                      57                     925
## OTU.2                       0                       1
## OTU.3                      19                     492
## OTU.4                       0                       0
## OTU.5                       0                       0
##       12C-Con_rep1_1.679431_3 12C-Con_rep1_1.682305_4
## OTU.1                    5809                   14645
## OTU.2                      73                    1077
## OTU.3                    5078                   21603
## OTU.4                       0                       0
## OTU.5                       3                      43
##       12C-Con_rep1_1.689408_5
## OTU.1                   15089
## OTU.2                    7499
## OTU.3                   36426
## OTU.4                      11
## OTU.5                    1017
physeq_rep3_t = OTU_qPCR_trans(physeq_rep3, physeq_rep3_qPCR)
phyloseq::otu_table(physeq_rep3_t) %>% .[1:5, 1:5]
## OTU Table:          [5 taxa and 5 samples]
##                      taxa are rows
##       12C-Con_rep1_1.668185_1 12C-Con_rep1_1.680254_2
## OTU.1               223148756               504984567
## OTU.2                       0                  545929
## OTU.3                74382919               268597197
## OTU.4                       0                       0
## OTU.5                       0                       0
##       12C-Con_rep1_1.679431_3 12C-Con_rep1_1.682305_4
## OTU.1               207913325               252808193
## OTU.2                 2612786                18591630
## OTU.3               181749675               372920136
## OTU.4                       0                       0
## OTU.5                  107375                  742284
##       12C-Con_rep1_1.689408_5
## OTU.1               327552736
## OTU.2               162788652
## OTU.3               790737356
## OTU.4                  238789
## OTU.5                22077085

BD shift (Z) and atom fraction excess (A)

With the transformed OTU abundance counts, let's calculate the BD shift (Z) and atom fraction excess (A) for each OTU. We'll be comparing controls (designated by the control_expr option) and the labeled treatment (all samples not identified by the control_expr expression). In this case the control_expr is set to Treatment=="12C-Con", which will select all samples where the Treatment column in the phyloseq sample_data table equals 12C-Con.

The treatment_rep option designates which column in the sample_data table defines the replicate gradients (eg., 12C-Con_rep1, 12C-Con_rep2, etc).

# The 'Treatment' column designates treatment vs control
# The 'Replicate' column designates treatment replicates
data.frame(sample_data(physeq_rep3)) %>%
  dplyr::select(Treatment, Replicate) %>%
  distinct
##   Treatment Replicate
## 1   12C-Con         1
## 2   12C-Con         2
## 3   12C-Con         3
## 4   13C-Glu         1
## 5   13C-Glu         2
## 6   13C-Glu         3
atomX = qSIP_atom_excess(physeq_rep3_t,
                         control_expr='Treatment=="12C-Con"',
                         treatment_rep='Replicate')
atomX %>% names
## [1] "W" "A"

The output is a list of data.frames. The W table lists the weighted mean BD for each OTU in each gradient. This can be considered a sort-of 'raw data' table. The real meat of the output is the A table, which lists (among other things) the BD shift (Z) and atom fraction excess (A) as defined by Hungate et al., (2015).

Note: a more complicated expression can be used to designate control samples. For example: 'Treatment == “12C-Con” & Day == 1'

Bootstrap confidence intervals

OK. The last step is to calculate atom % excess confidence intervals (CI). This is necessary for getting some sort of indication on how accurate the atom fraction excess estimations are. Also, Hungate et al., (2015) uses the CIs to call incorporators, in which an OTU was considered an incorporator if the CI was greater than, and did not span, zero.

We'll use 20 bootstrap replicates for this tutorial, but I recommend 100 for a real analysis. Bootstrap replicates can be run in parallel (see the function documentation).

df_atomX_boot = qSIP_bootstrap(atomX, n_boot=20)
df_atomX_boot %>% head
## # A tibble: 6 x 11
##   OTU    Wlab Wlight      Z    Gi Mlight Mheavymax  Mlab     A A_CI_low
##   <chr> <dbl>  <dbl>  <dbl> <dbl>  <dbl>     <dbl> <dbl> <dbl>    <dbl>
## 1 OTU.1  1.72   1.70 0.0222 0.636   308.      318.  312. 0.412   0.0541
## 2 OTU.2  1.71   1.69 0.0117 0.586   308.      318.  310. 0.218   0.119 
## 3 OTU.3  1.73   1.70 0.0347 0.608   308.      318.  314. 0.644   0.348 
## 4 OTU.4  1.73   1.70 0.0261 0.705   308.      318.  313. 0.484   0.408 
## 5 OTU.5  1.73   1.71 0.0242 0.710   308.      318.  312. 0.449   0.191 
## 6 OTU.6  1.73   1.70 0.0283 0.638   308.      318.  313. 0.526   0.407 
## # … with 1 more variable: A_CI_high <dbl>

Let's use the same threshold as Hungate et al., (2015) for defining incorporators.

CI_threshold = 0
df_atomX_boot = df_atomX_boot %>%
  mutate(Incorporator = A_CI_low > CI_threshold,
                OTU = reorder(OTU, -A))

How many incorporators?

n_incorp = df_atomX_boot %>%
  filter(Incorporator == TRUE) %>%
  nrow
cat('Number of incorporators:', n_incorp, '\n')
## Number of incorporators: 6

OK, let's plot the results.

ggplot(df_atomX_boot, aes(OTU, A, ymin=A_CI_low, ymax=A_CI_high, color=Incorporator)) +
  geom_pointrange(size=0.25) +
  geom_linerange() +
  geom_hline(yintercept=0, linetype='dashed', alpha=0.5) +
  labs(x='OTU', y='Atom fraction excess') +
  theme_bw() +
  theme(
    axis.text.x = element_blank()
  )

plot of chunk unnamed-chunk-12

Session info

sessionInfo()
## R version 3.4.3 (2017-11-30)
## Platform: x86_64-apple-darwin15.6.0 (64-bit)
## Running under: macOS Sierra 10.12.6
## 
## Matrix products: default
## BLAS: /Library/Frameworks/R.framework/Versions/3.4/Resources/lib/libRblas.0.dylib
## LAPACK: /Library/Frameworks/R.framework/Versions/3.4/Resources/lib/libRlapack.dylib
## 
## locale:
## [1] C/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8
## 
## attached base packages:
## [1] stats     graphics  grDevices utils     datasets  methods   base     
## 
## other attached packages:
## [1] phyloseq_1.22.3 HTSSIP_1.4.1    ggplot2_3.2.0   tidyr_0.8.3    
## [5] dplyr_0.8.0.1  
## 
## loaded via a namespace (and not attached):
##   [1] nlme_3.1-131               bitops_1.0-6              
##   [3] matrixStats_0.54.0         bit64_0.9-7               
##   [5] doParallel_1.0.14          RColorBrewer_1.1-2        
##   [7] GenomeInfoDb_1.14.0        tools_3.4.3               
##   [9] backports_1.1.4            utf8_1.1.4                
##  [11] R6_2.4.0                   coenocliner_0.2-2         
##  [13] vegan_2.5-1                rpart_4.1-15              
##  [15] Hmisc_4.2-0                DBI_1.0.0                 
##  [17] lazyeval_0.2.2             BiocGenerics_0.24.0       
##  [19] mgcv_1.8-28                colorspace_1.4-1          
##  [21] permute_0.9-5              ade4_1.7-13               
##  [23] nnet_7.3-12                withr_2.1.2               
##  [25] tidyselect_0.2.5           gridExtra_2.3             
##  [27] DESeq2_1.18.1              bit_1.1-14                
##  [29] compiler_3.4.3             cli_1.1.0                 
##  [31] Biobase_2.38.0             htmlTable_1.13.1          
##  [33] DelayedArray_0.4.1         labeling_0.3              
##  [35] scales_1.0.0               checkmate_1.9.3           
##  [37] genefilter_1.60.0          stringr_1.4.0             
##  [39] digest_0.6.19              foreign_0.8-71            
##  [41] XVector_0.18.0             base64enc_0.1-3           
##  [43] pkgconfig_2.0.2            htmltools_0.3.6           
##  [45] highr_0.8                  htmlwidgets_1.3           
##  [47] rlang_0.4.0                RSQLite_2.1.1             
##  [49] rstudioapi_0.10            jsonlite_1.6              
##  [51] BiocParallel_1.12.0        acepack_1.4.1             
##  [53] RCurl_1.95-4.12            magrittr_1.5              
##  [55] GenomeInfoDbData_1.0.0     Formula_1.2-3             
##  [57] biomformat_1.6.0           Matrix_1.2-17             
##  [59] fansi_0.4.0                Rcpp_1.0.1                
##  [61] munsell_0.5.0              S4Vectors_0.16.0          
##  [63] ape_5.3                    stringi_1.4.3             
##  [65] MASS_7.3-51.4              SummarizedExperiment_1.8.1
##  [67] zlibbioc_1.24.0            rhdf5_2.22.0              
##  [69] plyr_1.8.4                 blob_1.1.1                
##  [71] grid_3.4.3                 parallel_3.4.3            
##  [73] crayon_1.3.4               lattice_0.20-38           
##  [75] Biostrings_2.46.0          splines_3.4.3             
##  [77] multtest_2.34.0            annotate_1.56.2           
##  [79] locfit_1.5-9.1             zeallot_0.1.0             
##  [81] knitr_1.18                 pillar_1.4.1              
##  [83] igraph_1.2.4               GenomicRanges_1.30.3      
##  [85] markdown_0.9               geneplotter_1.56.0        
##  [87] reshape2_1.4.3             codetools_0.2-16          
##  [89] stats4_3.4.3               XML_3.98-1.19             
##  [91] glue_1.3.1                 evaluate_0.14             
##  [93] latticeExtra_0.6-28        data.table_1.10.4-3       
##  [95] vctrs_0.1.0                foreach_1.4.4             
##  [97] gtable_0.3.0               purrr_0.3.2               
##  [99] assertthat_0.2.1           mime_0.6                  
## [101] xtable_1.8-4               survival_2.44-1.1         
## [103] tibble_2.1.1               iterators_1.0.10          
## [105] memoise_1.1.0              AnnotationDbi_1.40.0      
## [107] IRanges_2.12.0             cluster_2.0.6