Jonathan W. Leff
2016-06-14
mctoolsr (prononounced M-C-tools-R) is an R package developed to facilitate microbial community analyses. The current functions are meant to handle an input taxa (OTU) table in either biom or tab-delimited format and help streamline common (and more specialized) downstream analyses. It is under active development, so please submit bug reports and feature requests as indicated below.
This document serves as a brief introduction to using mctoolsr. This document will go through getting the package working and a few examples using the most popular functions.
See here for a list of current functions: https://github.com/leffj/mctoolsr/blob/master/function_list.md
mctoolsr is available on Github at: https://github.com/leffj/mctoolsr
To install:
install.packages("devtools")
devtools::install_github("leffj/mctoolsr")
To use in a script, load and attach the package using:
library(mctoolsr)
mctoolsr is under active devlopment. Update regularly to use the latest features and for bug fixes. You will have to load and attach as above after updating.
devtools::install_github("leffj/mctoolsr")
https://github.com/leffj/mctoolsr/issues
The package is in initial development, so for now, please simply cite the Github URL: https://github.com/leffj/mctoolsr/
Note that the following examples use an example dataset taken from a study examining the bacterial communities associated with fruits and vegetables (Leff et al. 2013). You can find this dataset in the mctoolsr/examples directory.
You can load a taxa (i.e. OTU) table in biom format using the following approach. Note that you don't need to use system.file()
when using your own filepaths -- just use the filepath directly. This function just helps your system find the example files.
One of the nice things about loading your data this way is that all the sample IDs will be matched between your taxon table and metadata so that they will be in the same order and any sample IDs not present in one or the other will be dropped.
You can optionally filter out samples of a specific type during this step, but this can also be done separately as shown here.
tax_table_fp = system.file('extdata', 'fruits_veggies_taxa_table_wTax.biom',
package = 'mctoolsr')
map_fp = system.file('extdata', 'fruits_veggies_metadata.txt',
package = 'mctoolsr')
input = load_taxa_table(tax_table_fp, map_fp)
## 32 samples loaded
The loaded data will consist of three parts:
- The taxon table itself: "data_loaded"
- The metadata: "map_loaded"
- The taxonomic classifiers (if provided in the biom file): "taxonomy_loaded"
Any of these components can be quickly accessed using the '$' sign notation as shown in the next example.
This can be achieved simply by calculating column sums on the taxon table:
sort(colSums(input$data_loaded))
## ProA37 ProB70 ProC66 ProB39 ProC40 ProB57 ProA12 ProA58 ProB34 ProC36
## 1009 1011 1068 1152 1179 1192 1199 1216 1265 1313
## ProA65 ProB9 ProB60 ProC65 ProB35 ProB10 ProB67 ProA66 ProB36 ProB71
## 1367 1371 1395 1409 1492 1599 1611 1614 1642 1745
## ProB40 ProA36 ProB58 ProB33 ProA16 ProA35 ProA33 ProB12 ProA13 ProA34
## 1771 1802 1860 2257 2312 2530 2585 2642 2819 2982
## ProA15 ProA14
## 3291 3390
As you can see from the previous example, we can rarefy (i.e. normalize for variable sequence depths) to 1000 sequences per sample without losing any samples. This can be done using the following command:
input_rar = single_rarefy(input, 1000)
## 32 samples remaining
colSums(input_rar$data_loaded)
## ProA12 ProA13 ProA14 ProA15 ProA16 ProA33 ProA34 ProA35 ProA36 ProA37
## 1000 1000 1000 1000 1000 1000 1000 1000 1000 1000
## ProA58 ProA65 ProA66 ProB10 ProB12 ProB33 ProB34 ProB35 ProB36 ProB39
## 1000 1000 1000 1000 1000 1000 1000 1000 1000 1000
## ProB40 ProB57 ProB58 ProB60 ProB67 ProB70 ProB71 ProB9 ProC36 ProC40
## 1000 1000 1000 1000 1000 1000 1000 1000 1000 1000
## ProC65 ProC66
## 1000 1000
It is useful to get a feel for the taxonomic composition of your samples early on in the exploratory data analysis process. This can quickly be done by calculating taxonomic summaries at higher taxonomic levels - in this case at the phylum level. The values represent the sum of all the relative abundances for OTUs classified as belonging to the indicated phylum. In this example just the first few phyla and samples are shown.
tax_sum_phyla = summarize_taxonomy(input_rar, level = 2, report_higher_tax = FALSE)
tax_sum_phyla[1:5, 1:8]
## ProA12 ProA13 ProA14 ProA15 ProA16 ProA33 ProA34 ProA35
## p__ 0.001 0.000 0.000 0.000 0.000 0.000 0.000 0.000
## p__[Thermi] 0.000 0.000 0.000 0.000 0.000 0.000 0.003 0.000
## p__Acidobacteria 0.002 0.000 0.000 0.000 0.000 0.000 0.000 0.000
## p__Actinobacteria 0.380 0.006 0.002 0.024 0.002 0.051 0.075 0.087
## p__Armatimonadetes 0.000 0.000 0.000 0.003 0.000 0.000 0.000 0.000
You can also quickly show differences in the relative abundances of taxanomic groups using a heatmap as shown below. We can easily see that mushrooms have a much lower relative abundance of Enterobacteriaceae than the other sample types.
tax_sum_families = summarize_taxonomy(input_rar, level = 5, report_higher_tax = FALSE)
plot_ts_heatmap(tax_sum_families, input_rar$map_loaded, 0.01, 'Sample_type', custom_sample_order = c('Lettuce', 'Spinach', 'Strawberries', 'Mushrooms'))
For dissimilarity-based analyses such as ordinations and PERMANOVA, it is necessary to calculate a dissimilarity matrix. There is currently support for Bray-Curtis dissimilarities based on square-root transformed data. This is a widely used dissimilarity metric for these analyses, but others will be added as requested.
dm = calc_dm(input_rar$data_loaded)
There are two ways to plot ordinations in mctoolsr. The multistep way is shown here, but there is also a shortcut using the plot_nmds()
function.
ord = calc_ordination(dm, 'nmds')
## Run 0 stress 0.1206532
## Run 1 stress 0.1206548
## ... procrustes: rmse 0.0001599613 max resid 0.0007254419
## *** Solution reached
plot_ordination(input_rar, ord, 'Sample_type', 'Farm_type', hulls = TRUE)
It is easy to filter samples from your dataset in mctoolsr. You can specify to remove samples meeting a specified condition in the metadata or keep those samples. In the example below, lettuce samples are removed, and the ordination is plotted again.
input_rar_filt = filter_data(input_rar, 'Sample_type', filter_vals = 'Lettuce')
## 28 samples remaining
dm = calc_dm(input_rar_filt$data_loaded)
ord = calc_ordination(dm, 'nmds')
## Run 0 stress 0.1235166
## Run 1 stress 0.1212337
## ... New best solution
## ... procrustes: rmse 0.08556598 max resid 0.2920493
## Run 2 stress 0.1146686
## ... New best solution
## ... procrustes: rmse 0.08985927 max resid 0.2832726
## Run 3 stress 0.1145554
## ... New best solution
## ... procrustes: rmse 0.009807793 max resid 0.0367577
## Run 4 stress 0.1217904
## Run 5 stress 0.1145536
## ... New best solution
## ... procrustes: rmse 0.0002374229 max resid 0.0008392381
## *** Solution reached
plot_ordination(input_rar_filt, ord, 'Sample_type', 'Farm_type', hulls = TRUE)
There are multiple taxa filtering options in mctoolsr. This example shows how to explore the proteobacteria sequences across the samples. Taxa can also be filtered based on their relative abundance.
input_proteobact = filter_taxa_from_input(input, taxa_to_keep = 'p__Proteobacteria')
## 2265 taxa removed
sort(colSums(input_proteobact$data_loaded))
## ProC40 ProA14 ProA12 ProA58 ProA16 ProB70 ProB10 ProB9 ProA37 ProB12
## 231 435 459 614 780 843 889 940 980 1019
## ProC66 ProC36 ProB57 ProA13 ProA36 ProB39 ProB34 ProB60 ProA65 ProA35
## 1032 1051 1063 1080 1082 1082 1196 1196 1261 1336
## ProC65 ProB71 ProA34 ProB67 ProA66 ProB35 ProB58 ProB36 ProA15 ProB40
## 1369 1376 1406 1422 1436 1472 1544 1594 1601 1748
## ProA33 ProB33
## 1768 2126
input_proteobact_rar = single_rarefy(input_proteobact, 219)
## 32 samples remaining
plot_nmds(calc_dm(input_proteobact_rar$data_loaded), metadata_map = input_proteobact_rar$map_loaded,
color_cat = 'Sample_type')
## Run 0 stress 0.123787
## Run 1 stress 0.125684
## Run 2 stress 0.1144185
## ... New best solution
## ... procrustes: rmse 0.09762337 max resid 0.2789194
## Run 3 stress 0.1252272
## Run 4 stress 0.114415
## ... New best solution
## ... procrustes: rmse 0.001719574 max resid 0.005385391
## *** Solution reached
It is often useful to determine the taxa driving differences between the community compositions of different sample types. This example shows one way to do this to determine taxa driving differences between sample types.
tax_sum_families = summarize_taxonomy(input_rar_filt, level = 5, report_higher_tax = FALSE)
taxa_summary_by_sample_type(tax_sum_families, input_rar_filt$map_loaded,
type_header = 'Sample_type', filter_level = 0.05, test_type = 'KW')
## pvals pvalsBon pvalsFDR Mushrooms
## f__Pseudomonadaceae 2.251673e-05 0.0001576171 0.0001576171 0.141625
## f__Sphingobacteriaceae 7.934777e-05 0.0005554344 0.0002777172 0.293125
## f__[Weeksellaceae] 3.012755e-04 0.0021089282 0.0007029761 0.105625
## f__Enterobacteriaceae 3.120009e-04 0.0021840065 0.0005460016 0.033250
## unclassified 1.977469e-03 0.0138422830 0.0027684566 0.077125
## f__Bacillaceae 2.342285e-02 0.1639599761 0.0273266627 0.002250
## f__Sphingomonadaceae 3.513676e-02 0.2459572979 0.0351367568 0.008000
## Spinach Strawberries
## f__Pseudomonadaceae 0.0678571429 0.0019230769
## f__Sphingobacteriaceae 0.0032857143 0.0007692308
## f__[Weeksellaceae] 0.0021428571 0.0016153846
## f__Enterobacteriaceae 0.7261428571 0.6177692308
## unclassified 0.0301428571 0.0226923077
## f__Bacillaceae 0.0042857143 0.1580769231
## f__Sphingomonadaceae 0.0005714286 0.0611538462
This analysis demonstrates that Pseudomonadaceae and Sphingobacteriaceae tend to have higher relative abundances on mushrooms than spinach and strawberries. The p values are based on Kruskal-Wallis tests and two different corrections are reported to deal with the multiple comparisons (Bonferroni and FDR). Rare taxa are filtered out using the filter_level
peramter. The values indicated under the sample types are mean relative abundances.
Sometimes it is necessary to calculate mean dissimilarities. This is important in cases where sample types are pseudoreplecated. This is not the case here, but this example demonstrates this functionality.
dm = calc_dm(input_rar_filt$data_loaded)
dm_aggregated = calc_mean_dissimilarities(dm, input_rar_filt$map_loaded,
'Sample_Farming', return_map = TRUE)
ord = calc_ordination(dm_aggregated$dm, ord_type = 'nmds')
## Run 0 stress 0
## Run 1 stress 2.739166e-05
## ... procrustes: rmse 0.218308 max resid 0.3745
## Run 2 stress 0
## ... procrustes: rmse 0.2446559 max resid 0.4250695
## Run 3 stress 0
## ... procrustes: rmse 0.1148544 max resid 0.2019954
## Run 4 stress 9.070047e-05
## ... procrustes: rmse 0.01741263 max resid 0.03149269
## Run 5 stress 0
## ... procrustes: rmse 0.0294611 max resid 0.05420394
## Run 6 stress 8.900239e-05
## ... procrustes: rmse 0.01434851 max resid 0.02136145
## Run 7 stress 0
## ... procrustes: rmse 0.1967323 max resid 0.3594271
## Run 8 stress 0
## ... procrustes: rmse 0.1852564 max resid 0.3277199
## Run 9 stress 0
## ... procrustes: rmse 0.2321982 max resid 0.370431
## Run 10 stress 0
## ... procrustes: rmse 0.2616707 max resid 0.4185312
## Run 11 stress 0
## ... procrustes: rmse 0.2533339 max resid 0.4033764
## Run 12 stress 0
## ... procrustes: rmse 0.248833 max resid 0.3975527
## Run 13 stress 0.1842128
## Run 14 stress 0
## ... procrustes: rmse 0.0169415 max resid 0.02214461
## Run 15 stress 8.698707e-05
## ... procrustes: rmse 0.01545266 max resid 0.02704799
## Run 16 stress 8.18449e-05
## ... procrustes: rmse 0.246017 max resid 0.3454335
## Run 17 stress 8.648759e-05
## ... procrustes: rmse 0.1811352 max resid 0.3231973
## Run 18 stress 3.307416e-05
## ... procrustes: rmse 0.0169743 max resid 0.02833947
## Run 19 stress 0
## ... procrustes: rmse 0.2408064 max resid 0.3820194
## Run 20 stress 0
## ... procrustes: rmse 0.2072957 max resid 0.366378
## Warning in vegan::metaMDS(dm, k = 2): Stress is (nearly) zero - you may
## have insufficient data
plot_ordination(dm_aggregated, ord, color_cat = 'Sample_type')
Taxa tables that have been edited in *mctoolsr can be exported in text (tab delimited) format for later use. Use this function:
export_taxa_table(input_rar_filt, "export/path/otu_table_export.txt")