tidypopgen has two key functions to examine the quality of data
across loci and across individuals: qc_report_loci and
qc_report_indiv. This vignette uses a simulated data set to
illustrate these methods of data cleaning.
library(tidypopgen)
data <- gen_tibble(
system.file("extdata/related/families.bed",
package = "tidypopgen"
),
quiet = TRUE, backingfile = tempfile(),
valid_alleles = c("1", "2")
)#Quality control for individuals
## het_obs missingness
## Min. :0.3467 Min. :0.03018
## 1st Qu.:0.3736 1st Qu.:0.03642
## Median :0.3790 Median :0.03798
## Mean :0.3799 Mean :0.03867
## 3rd Qu.:0.3901 3rd Qu.:0.04110
## Max. :0.4015 Max. :0.04579
The output of qc_report_indiv supplies observed
heterozygosity per individual, and rate of missingness per individual as
standard.
These data can also be visualised using autoplot:
Here, the red line indicates a threshold for proportion of missing
loci, which is set as 5% by default, and can be altered using the
miss_threshold argument. Similarly, the blue lines are
drawn at 2 (inner) and 3 (outer) standard deviations from the mean
observed heterozygosity. These thresholds can be used to visualise
outliers and consider how to filter datasets.
We can see above that most individuals have low missingness, none are
above the default 5% threshold. However, if we wanted to filter
individuals to remove those with more than 4.5% of their genotypes
missing, we can use filter.
## [1] 10
And if we wanted to remove outliers with particularly high or low
heterozygosity, we can again do so by using filter. As an
example, here we remove observations that lie more than 2 standard
deviations from the mean.
mean_val <- mean(individual_report$het_obs)
sd_val <- stats::sd(individual_report$het_obs)
lower <- mean_val - 2 * (sd_val)
upper <- mean_val + 2 * (sd_val)
data <- data %>% filter(indiv_het_obs(genotypes) > lower)
data <- data %>% filter(indiv_het_obs(genotypes) < upper)
nrow(data)## [1] 9
Next, we can look at relatedness within our sample. If the parameter
kings_threshold is provided to
qc_report_indiv(), then the report also calculates a KING
coefficient of relatedness matrix using the sample. The
kings_threshold is used to provide an output of the largest
possible group with no related individuals in the third column
to_keep. This boolean column recommends which individuals
to remove (FALSE) and to keep (TRUE) to achieve an unrelated sample.
## het_obs missingness to_keep id
## Min. :0.3688 Min. :0.03018 Mode :logical Length:9
## 1st Qu.:0.3774 1st Qu.:0.03642 FALSE:1 Class :character
## Median :0.3795 Median :0.03746 TRUE :8 Mode :character
## Mean :0.3851 Mean :0.03735
## 3rd Qu.:0.3985 3rd Qu.:0.04058
## Max. :0.4015 Max. :0.04266
We can remove the recommended individuals by using:
We can now view a summary of our cleaned data set again, showing that our data has reduced from 12 to 8 individuals.
## id genotypes
## Length:8 Length:8
## Class :character Class :character
## Mode :character Mode :character
## This gen_tibble is not grouped. For Hardy-Weinberg equilibrium, `qc_report_loci()` will assume individuals are part of the same population and HWE test p-values will be calculated across all individuals. If you wish to calculate HWE p-values within populations or groups, please use`group_by()` before calling `qc_report_loci()`.
## snp_id maf missingness hwe_p
## Length:961 Min. :0.0000 Min. :0.00000 Min. :0.00272
## Class :character 1st Qu.:0.1667 1st Qu.:0.00000 1st Qu.:0.32867
## Mode :character Median :0.2500 Median :0.00000 Median :0.53333
## Mean :0.2661 Mean :0.03733 Mean :0.50321
## 3rd Qu.:0.3750 3rd Qu.:0.12500 3rd Qu.:0.69231
## Max. :0.5000 Max. :0.37500 Max. :0.76503
The output of qc_report_loci supplies minor allele
frequency, rate of missingness, and a Hardy-Weinberg exact p-value for
each SNP. These data can be visualised in autoplot :
## Warning: `aes_string()` was deprecated in ggplot2 3.0.0.
## ℹ Please use tidy evaluation idioms with `aes()`.
## ℹ See also `vignette("ggplot2-in-packages")` for more information.
## ℹ The deprecated feature was likely used in the UpSetR package.
## Please report the issue to the authors.
## This warning is displayed once every 8 hours.
## Call `lifecycle::last_lifecycle_warnings()` to see where this warning was
## generated.
## Warning: Using `size` aesthetic for lines was deprecated in ggplot2 3.4.0.
## ℹ Please use `linewidth` instead.
## ℹ The deprecated feature was likely used in the UpSetR package.
## Please report the issue to the authors.
## This warning is displayed once every 8 hours.
## Call `lifecycle::last_lifecycle_warnings()` to see where this warning was
## generated.
## Warning: The `size` argument of `element_line()` is deprecated as of ggplot2 3.4.0.
## ℹ Please use the `linewidth` argument instead.
## ℹ The deprecated feature was likely used in the UpSetR package.
## Please report the issue to the authors.
## This warning is displayed once every 8 hours.
## Call `lifecycle::last_lifecycle_warnings()` to see where this warning was
## generated.
Using ‘overview’ provides an Upset plot, which is designed to show the intersection of different sets in the same way as a Venn diagram. SNPs can be divided into ‘sets’ that each pass predefined quality control threshold; a set of SNPs with missingness under a given threshold, a set of SNPs with MAF above a given threshold, and a set of SNPs with a Hardy-Weinberg exact p-value that falls above a given significance level.
The thresholds for each parameter, (percentage of missingness that is accepted, minor allele frequency cutoff, and Hardy-Weinberg equilibrium p-value) can be adjusted using the parameters provided in autoplot. For example:
autoplot(loci_report,
type = "overview",
miss_threshold = 0.03,
maf_threshold = 0.02,
hwe_p = 0.01
)The upset plot then visualises our 961 SNPs within their respective sets. The number above the second bar indicates that 262 SNPs occur in all 3 sets, meaning 262 SNPs pass all of our QC thresholds. The combined total of the first and second bars represents the number of SNPs that pass our MAF and HWE thresholds, here 939 SNPs.
To examine each QC measure in further detail, we can plot a different summary panel.
We can then begin to consider how to quality control this raw data set. Let’s start by filtering SNPs according to their minor allele frequency. We can visualise the MAF distribution using:
Here we can see there are some monomorphic SNPs in the data set.
Let’s filter out loci with a minor allele frequency lower than 2%, by
using select_loci_if. Here, we select all SNPs with a MAF
greater than 2%. This operation is equivalent to plink –maf 0.02.
## [1] 931
Following this, we can remove SNPs with a high rate of missingness. Lets say we want to remove SNPs that are missing in more than 5% of individuals, equivalent to using plink –geno 0.05
We can see here that most SNPs have low missingness, under our 5%
threshold, some do, however, have missingness over our threshold. To
remove these SNPs, we can again use select_loci_if.
## [1] 698
Finally, we may want to remove SNPs that show significant deviation from Hardy-Weinberg equilibrium, if our study design requires. To visualise SNPs with significant p-values in the Hardy-Weinberg exact test, we can again call autoplot:
None of the SNPs in our data are significant, however there may be circumstances where we would want to cut out the most extreme cases, if these data were real, these cases could indicate genotyping errors.
## [1] 697
For further analyses, it may be necessary to control for linkage in the data set. tidypopgen provides LD clumping. This option is similar to the –indep-pairwise flag in plink, but results in a more even distribution of loci when compared to LD pruning.
To explore why clumping is preferable to pruning, see https://privefl.github.io/bigsnpr/articles/pruning-vs-clumping.html
LD clumping requires a data set with no missingness. This means we
need to create an imputed data set before LD pruning, which we can do
quickly with gt_impute_simple.
Because we have removed individuals through our filtering, we first need to update the backingfiles with:
## Genetic distances are not sorted, setting them to zero
##
## gen_backing files updated, now
## using FBM RDS: /tmp/RtmpeA9m64/file2de14264ac25_v2.rds
## with FBM backing file: /tmp/RtmpeA9m64/file2de14264ac25_v2.bk
## make sure that you do NOT delete those files!
And then we can impute using:
In this example, if we want to remove SNPs with a correlation greater
than 0.2 in windows of 10 SNPs at a time, we can set these parameters
with thr_r2 and size respectively.
## [1] FALSE FALSE FALSE FALSE FALSE FALSE
loci_ld_clump provides a boolean vector the same length
as our list of SNPs, telling us which to keep in the data set. We can
then use this list to create a pruned version of our data:
The benefit of operating on a gen_tibble is that each
quality control step can be observed visually, and easily reversed if
necessary.
When we are happy with the quality of our data, we can create and
save a final quality controlled version of our gen_tibble
using gt_save.
##
## gen_tibble saved to /tmp/RtmpeA9m64/file2de1960b661.gt
## using FBM RDS: /tmp/RtmpeA9m64/file2de14264ac25_v2.rds
## with FBM backing file: /tmp/RtmpeA9m64/file2de14264ac25_v2.bk
## make sure that you do NOT delete those files!
## to reload the gen_tibble in another session, use:
## gt_load('/tmp/RtmpeA9m64/file2de1960b661.gt')
## [1] "/tmp/RtmpeA9m64/file2de1960b661.gt"
## [2] "/tmp/RtmpeA9m64/file2de14264ac25_v2.rds"
## [3] "/tmp/RtmpeA9m64/file2de14264ac25_v2.bk"
For some quality control measures, if you have a gen_tibble that
includes multiple datasets you may want to group by population before
running the quality control. This can be done using
group_by.
First, lets add some imaginary population data to our gen_tibble:
We can then group by population and run quality control on each group:
## # A tibble: 6 × 4
## snp_id maf missingness hwe_p
## <chr> <dbl> <dbl> <dbl>
## 1 2 0.438 0 0.429
## 2 3 0.188 0 1
## 3 5 0.25 0 1
## 4 7 0.312 0 0.429
## 5 8 0.312 0 1.14
## 6 10 0.438 0 0.429
The loci report that we receive here will calculate Hardy-Weinberg equilibrium for each SNP within each population separately, providing a Bonferroni corrected p-value for each SNP.
Similarly, we can run a quality control report for individuals within each population:
grouped_individual_report <- data %>%
group_by(population) %>%
qc_report_indiv(kings_threshold = 0.177)
head(grouped_individual_report)## # A tibble: 6 × 5
## het_obs missingness id group to_keep
## <dbl> <dbl> <chr> <chr> <lgl>
## 1 0.382 0 1 A TRUE
## 2 0.377 0 4 A TRUE
## 3 0.393 0 5 A TRUE
## 4 0.382 0 6 A TRUE
## 5 0.403 0 7 B TRUE
## 6 0.382 0 8 B TRUE
This is important when we have related individuals, as background population structure can affect the filtering of relatives.
It is also possible to run loci and indiv
functions on grouped data. This is useful when you want to run the same
quality control on each group of data, but don’t want to split the data
into separate gen_tibbles.
Grouped functions are built for efficiency and surpass the use of
applying a function with group_map.