This vignette provides a description of how to use the GENESIS package to analyze sequence data. We demonstrate the use of mixed models for genetic association testing, as PC-AiR PCs can be used as fixed effect covariates to adjust for population stratification, and a kinship matrix (or genetic relationship matrix) estimated from PC-Relate can be used to account for phenotype correlation due to genetic similarity among samples. To illustrate the methods, we use a small subset of data from 1000 Genomes Phase 3.
The first step is to convert a VCF file into the GDS file format used by GENESIS. We use the SeqArray package, which defines the extended GDS format used to capture all data in a VCF file. If the VCF files are split by chromosome, they can be combined into a single GDS file.
library(SeqArray)
vcffile <- system.file("extdata", "1KG",
paste0("1KG_phase3_subset_chr", 1:22, ".vcf.gz"),
package="GENESIS")
gdsfile <- tempfile()
seqVCF2GDS(vcffile, gdsfile, verbose=FALSE)
gds <- seqOpen(gdsfile)
gds
## Object of class "SeqVarGDSClass"
## File: /tmp/RtmpZy4WQH/file108f3e5c415a7a (419.7K)
## + [ ] *
## |--+ description [ ] *
## |--+ sample.id { Str8 100 LZMA_ra(37.8%), 309B } *
## |--+ variant.id { Int32 24639 LZMA_ra(7.99%), 7.7K } *
## |--+ position { Int32 24639 LZMA_ra(71.8%), 69.1K } *
## |--+ chromosome { Str8 24639 LZMA_ra(0.36%), 237B } *
## |--+ allele { Str8 24639 LZMA_ra(19.2%), 20.0K } *
## |--+ genotype [ ] *
## | |--+ data { Bit2 2x100x24657 LZMA_ra(18.7%), 224.9K } *
## | |--+ extra.index { Int32 3x0 LZMA_ra, 18B } *
## | \--+ extra { Int16 0 LZMA_ra, 18B }
## |--+ phase [ ]
## | |--+ data { Bit1 100x24639 LZMA_ra(0.06%), 201B } *
## | |--+ extra.index { Int32 3x0 LZMA_ra, 18B } *
## | \--+ extra { Bit1 0 LZMA_ra, 18B }
## |--+ annotation [ ]
## | |--+ id { Str8 24639 LZMA_ra(37.3%), 87.8K } *
## | |--+ qual { Float32 24639 LZMA_ra(0.17%), 173B } *
## | |--+ filter { Int32,factor 24639 LZMA_ra(0.17%), 173B } *
## | |--+ info [ ]
## | \--+ format [ ]
## \--+ sample.annotation [ ]
Next, we combine the GDS file with information about the samples, which we store in an AnnotatedDataFrame
(defined in the Biobase package). An AnnotatedDataFrame
combines a data.frame
with metadata describing each column. A SeqVarData
object (defined in the SeqVarTools package), contains both an open GDS file and an AnnotatedDataFrame
describing the samples. The sample.id
column in the AnnotatedDataFrame
must match the sample.id
node in the GDS file.
library(GENESIS)
library(Biobase)
library(SeqVarTools)
data(sample_annotation_1KG)
annot <- sample_annotation_1KG
head(annot)
## sample.id Population sex
## 1 HG00110 GBR F
## 2 HG00116 GBR M
## 3 HG00120 GBR F
## 4 HG00128 GBR F
## 5 HG00136 GBR M
## 6 HG00137 GBR F
# simulate some phenotype data
set.seed(4)
annot$outcome <- rnorm(nrow(annot))
metadata <- data.frame(labelDescription=c("sample id",
"1000 genomes population",
"sex",
"simulated phenotype"),
row.names=names(annot))
annot <- AnnotatedDataFrame(annot, metadata)
all.equal(annot$sample.id, seqGetData(gds, "sample.id"))
## [1] TRUE
seqData <- SeqVarData(gds, sampleData=annot)
The first step for association testing is to fit the model under the null hypothesis that each SNP has no effect. This null model contains all of the covariates, including ancestry representative PCs, as well as any random effects, such as a polygenic effect due to genetic relatedness, but it does not include any SNP genotype terms as fixed effects.
The type of model fit depends on the arguments to fitNullModel
. Including a cov.mat
argument will result in a mixed model, while omitting this argument will run a standard linear model. A logistic model is specified with family="binomial"
. In the case of a logistic model and a covariance matrix, fitNullModel
will use the GMMAT algorithm. Including a group.var
argument will allow heteroscedastic variance (for linear models or linear mixed models only).
# add PCs to sample annotation in SeqVarData object
annot <- AnnotatedDataFrame(pc.df)
sampleData(seqData) <- annot
# covariance matrix from pcrelate output
grm <- pcrelateToMatrix(pcrel, scaleKin=2)
# fit the null model
nullmod <- fitNullModel(seqData, outcome="outcome",
covars=c("sex", "Population", paste0("PC", 1:2)),
cov.mat=grm, verbose=FALSE)
To run a test using the null model, we first create an iterator object specifying how we want variants to be selected. (See the documentation for the SeqVarIterator
class in SeqVarTools for more details.) For single-variant tests (GWAS), it is common to use a block iterator that reads variants in blocks (default is 10,000 variants per block).
For example purposes, we restrict our analysis to chromosome 1. The seqSetFilter
function can be used to restrict the set of variants tested in other ways (e.g., variants that pass a quality filter).
# select chromosome 1
seqSetFilterChrom(seqData, include=1)
## # of selected variants: 1,120
iterator <- SeqVarBlockIterator(seqData, verbose=FALSE)
assoc <- assocTestSingle(iterator, nullmod, verbose=FALSE,
BPPARAM=BiocParallel::SerialParam())
head(assoc)
## variant.id chr pos allele.index n.obs freq MAC Score Score.SE
## 1 1 1 828740 1 100 0.035 7 0.4554795 2.888007
## 2 2 1 913272 1 100 0.010 2 -0.2775530 1.403110
## 3 3 1 1171878 1 100 0.005 1 0.2227947 1.015806
## 4 4 1 1242288 1 100 0.025 5 -1.2206682 2.163235
## 5 5 1 1378837 1 100 0.670 66 -11.2824035 6.080219
## 6 6 1 1403820 1 100 0.015 3 2.3221454 1.692363
## Score.Stat Score.pval Est Est.SE PVE
## 1 0.1577141 0.8746821 0.05461001 0.3462595 0.0002926286
## 2 -0.1978127 0.8431916 -0.14098160 0.7127024 0.0004603457
## 3 0.2193280 0.8263945 0.21591530 0.9844400 0.0005659318
## 4 -0.5642791 0.5725642 -0.26084970 0.4622707 0.0037459651
## 5 -1.8555915 0.0635118 -0.30518497 0.1644678 0.0405079734
## 6 1.3721322 0.1700223 0.81077907 0.5908899 0.0221496917
The default test is a Score test, but the Wald test is also available for continuous outcomes.
If there are multiallelic variants, each alternate allele is tested separately. The allele.index
column in the output differentiates between different alternate alleles for the same variant.
We make a QQ plot to examine the results.
qqPlot <- function(pval) {
pval <- pval[!is.na(pval)]
n <- length(pval)
x <- 1:n
dat <- data.frame(obs=sort(pval),
exp=x/n,
upper=qbeta(0.025, x, rev(x)),
lower=qbeta(0.975, x, rev(x)))
ggplot(dat, aes(-log10(exp), -log10(obs))) +
geom_line(aes(-log10(exp), -log10(upper)), color="gray") +
geom_line(aes(-log10(exp), -log10(lower)), color="gray") +
geom_point() +
geom_abline(intercept=0, slope=1, color="red") +
xlab(expression(paste(-log[10], "(expected P)"))) +
ylab(expression(paste(-log[10], "(observed P)"))) +
theme_bw()
}
qqPlot(assoc$Score.pval)
We can aggregate rare variants for association testing to decrease multiple testing burden and increase statistical power. We can create functionally agnostic units using a SeqVarWindowIterator
. This iterator type generates a sliding window over the genome, with user-specified width and step size. We can also create units with specific start and end points or containing specific variants, using a SeqVarRangeIterator
or a SeqVarListIterator
.
In this example, we illustrate defining ranges based on known genes. We run a burden test, setting a maximum alternate allele frequency to exclude common variants.
library(GenomicRanges)
library(TxDb.Hsapiens.UCSC.hg19.knownGene)
## Warning: replacing previous import 'utils::findMatches' by
## 'S4Vectors::findMatches' when loading 'AnnotationDbi'
# return the variants on chromosome 1 as a GRanges object
seqSetFilterChrom(seqData, include=1)
## # of selected variants: 1,120
gr <- granges(gds)
# find variants that overlap with each gene
txdb <- TxDb.Hsapiens.UCSC.hg19.knownGene
gr <- renameSeqlevels(gr, paste0("chr", seqlevels(gr)))
ts <- transcriptsByOverlaps(txdb, gr, columns="GENEID")
# simplistic example - define genes as overlapping transcripts
genes <- reduce(ts)
genes <- renameSeqlevels(genes, sub("chr", "", seqlevels(genes)))
# create an iterator where each successive unit is a different gene
iterator <- SeqVarRangeIterator(seqData, variantRanges=genes, verbose=FALSE)
# do a burden test on the rare variants in each gene
assoc <- assocTestAggregate(iterator, nullmod, AF.max=0.05, test="Burden",
BPPARAM=BiocParallel::SerialParam(), verbose=FALSE)
The output of an aggregate test is a list with two elements: 1) a data.frame with the test results for each aggregate unit, and 2) a list of data.frames containing the variants in each aggregate unit.
head(assoc$results)
## n.site n.alt n.sample.alt Score Score.SE Score.Stat Score.pval
## 1 1 3 3 2.3221454 1.692363 1.3721322 0.17002228
## 2 1 5 5 4.0027945 2.246167 1.7820556 0.07474017
## 3 1 1 1 -0.6661171 1.043845 -0.6381379 0.52338391
## 4 1 9 9 -1.8242825 2.581275 -0.7067371 0.47972988
## 5 0 0 0 NA NA NA NA
## 6 1 1 1 -0.4031965 1.023646 -0.3938825 0.69366776
## Est Est.SE PVE
## 1 0.8107791 0.5908899 0.022149692
## 2 0.7933763 0.4452029 0.037360979
## 3 -0.6113339 0.9579965 0.004790765
## 4 -0.2737938 0.3874055 0.005876132
## 5 NA NA NA
## 6 -0.3847838 0.9768997 0.001825195
head(assoc$variantInfo)
## [[1]]
## variant.id chr pos allele.index n.obs freq MAC weight
## 1 6 1 1403820 1 100 0.015 3 1
##
## [[2]]
## variant.id chr pos allele.index n.obs freq MAC weight
## 1 7 1 1421285 1 100 0.025 5 1
##
## [[3]]
## variant.id chr pos allele.index n.obs freq MAC weight
## 1 12 1 2023475 1 100 0.005 1 1
##
## [[4]]
## variant.id chr pos allele.index n.obs freq MAC weight
## 1 21 1 3254100 1 100 0.045 9 1
##
## [[5]]
## [1] variant.id chr pos allele.index n.obs
## [6] freq MAC weight
## <0 rows> (or 0-length row.names)
##
## [[6]]
## variant.id chr pos allele.index n.obs freq MAC weight
## 1 24 1 3818550 1 100 0.005 1 1
Conomos M.P., Reiner A.P., Weir B.S., & Thornton T.A. (2016). Model-free Estimation of Recent Genetic Relatedness. American Journal of Human Genetics, 98(1), 127-148.
Conomos M.P., Miller M.B., & Thornton T.A. (2015). Robust Inference of Population Structure for Ancestry Prediction and Correction of Stratification in the Presence of Relatedness. Genetic Epidemiology, 39(4), 276-293.
Manichaikul, A., Mychaleckyj, J.C., Rich, S.S., Daly, K., Sale, M., & Chen, W.M. (2010). Robust relationship inference in genome-wide association studies. Bioinformatics, 26(22), 2867-2873.
Breslow NE and Clayton DG. (1993). Approximate Inference in Generalized Linear Mixed Models. Journal of the American Statistical Association 88: 9-25.
Chen H, Wang C, Conomos MP, Stilp AM, Li Z, Sofer T, Szpiro AA, Chen W, Brehm JM, Celedon JC, Redline S, Papanicolaou GJ, Thornton TA, Laurie CC, Rice K and Lin X. Control for Population Structure and Relatedness for Binary Traits in Genetic Association Studies Using Logistic Mixed Models. American Journal of Human Genetics, 98(4): 653-66.
Leal, S.M. & Li, B. (2008). Methods for Detecting Associations with Rare Variants for Common Diseases: Application to Analysis of Sequence Data. American Journal of Human Genetics, 83(3), 311-321.