This vignette demonstrates how to manually collect GO gene sets from BioMart. The source code for retrieving GO gene sets for all supported organisms on BioMart can be found from the following location:

system.file("scripts", "biomart_genesets.R", package = "BioMartGOGeneSets")
## [1] "/private/var/folders/g3/f2y6rp510nxf3t5sj6h902bc0000gr/T/Rtmpd7g1fJ/temp_libpath9782166ad12d/BioMartGOGeneSets/scripts/biomart_genesets.R"

Retrieving GO gene sets from BioMart is a two-step process.

Step1. Retrieve the relations between GO terms and genes for a specific organism

The relations can be obtained from BioMart with the R package biomaRt. First let’s create a “Mart” object which is like a connection to the BioMart web-service. Because this is a gene-related relation, we set argument biomart to "genes".

library(biomaRt)
## Warning: package 'biomaRt' was built under R version 4.2.1
ensembl = useEnsembl(biomart = "genes")
ensembl
## Object of class 'Mart':
##   Using the ENSEMBL_MART_ENSEMBL BioMart database
##   No dataset selected.

Please note, if later when you use biomaRt and see errors of “timeout”, you can select a different mirror, such as:

useEnsembl(biomart = "genes", mirror = "uswest")

Next we need to select an organism. In BioMart, it is called a “dataset”. To get a proper value of an organism dataset, we can use listDatasets() to get a list of supported datasets.

datasets = listDatasets(ensembl)
dim(datasets)
## [1] 213   3
head(datasets)
##                        dataset                           description
## 1 abrachyrhynchus_gene_ensembl Pink-footed goose genes (ASM259213v1)
## 2     acalliptera_gene_ensembl      Eastern happy genes (fAstCal1.2)
## 3   acarolinensis_gene_ensembl       Green anole genes (AnoCar2.0v2)
## 4    acchrysaetos_gene_ensembl       Golden eagle genes (bAquChr1.2)
## 5    acitrinellus_gene_ensembl        Midas cichlid genes (Midas_v5)
## 6    amelanoleuca_gene_ensembl       Giant panda genes (ASM200744v2)
##       version
## 1 ASM259213v1
## 2  fAstCal1.2
## 3 AnoCar2.0v2
## 4  bAquChr1.2
## 5    Midas_v5
## 6 ASM200744v2

You can see there are a huge number of organisms supported in BioMart. The first column in datasets contains valid values for dataset which will be used in later functions.

Sometimes it is not easy to find the dataset of your organism, especially when the organism is not a model organism. The second column in datasets contains dataset descriptions, which sometimes can be helpful to find the right one.

In this example, we use the dataset for human "hsapiens_gene_ensembl".

To use the dataset, we specify dataset in the function useDateset(), which is like adding a flag of which dataset to use.

ensembl = useDataset(dataset = "hsapiens_gene_ensembl", mart = ensembl)
ensembl
## Object of class 'Mart':
##   Using the ENSEMBL_MART_ENSEMBL BioMart database
##   Using the hsapiens_gene_ensembl dataset

The dataset is like a giant table with a huge number of columns which provide massive additional information for genes. Here we are only interested in GO-related information. In the dataset, the table columns are called “attributes”. There are a huge number of supported attributes in a dataset. The complete list of attributes can be obtained by the function listAttributes().

all_at = listAttributes(mart = ensembl)
dim(all_at)
## [1] 3156    3
head(all_at)
##                            name                  description         page
## 1               ensembl_gene_id               Gene stable ID feature_page
## 2       ensembl_gene_id_version       Gene stable ID version feature_page
## 3         ensembl_transcript_id         Transcript stable ID feature_page
## 4 ensembl_transcript_id_version Transcript stable ID version feature_page
## 5            ensembl_peptide_id            Protein stable ID feature_page
## 6    ensembl_peptide_id_version    Protein stable ID version feature_page

To get proper values for the attributes, we need to go through the long table and sometimes this is not an easy task. The three attributes of GO-gene relations are c("ensembl_gene_id", "go_id", "namespace_1003"). Now we can use the function getBM() to obtain the GO-gene relation table.

at = c("ensembl_gene_id", "go_id", "namespace_1003")
go = getBM(attributes = at, mart = ensembl)

Check the first several rows in go:

head(go)
##   ensembl_gene_id      go_id     namespace_1003
## 1 ENSG00000210049 GO:0030533 molecular_function
## 2 ENSG00000210049 GO:0006412 biological_process
## 3 ENSG00000211459 GO:0003735 molecular_function
## 4 ENSG00000211459 GO:0005840 cellular_component
## 5 ENSG00000210077                              
## 6 ENSG00000210082 GO:0003735 molecular_function

For some organisms, the returned table might be huge, or due to the bad internet connection, the data retrieving may exceed the 5 min limit on BioMart, and you might see the following error:

Error in curl::curl_fetch_memory(url, handle = handle) :
  Timeout was reached: [uswest.ensembl.org:443] Operation timed out after 300005 milliseconds with 2966434 bytes received

In this case, you might need to split the query into blocks and make sure each job query is small.

genes = getBM(attributes = "ensembl_gene_id", mart = ensembl)[, 1]
go1 = getBM(attributes = at, mart = ensembl, filter = "ensembl_gene_id", value = genes[1:1000])
go2 = getBM(attributes = at, mart = ensembl, filter = "ensembl_gene_id", value = genes[1001:2000])
...
rbind(go1, go2, ...)

In this example, we will only demonstrate the Biological Process Ontology, and we convert the data frame go to a list of genes.

go = go[go$namespace_1003 == "biological_process", , drop = FALSE]
gs = split(go$ensembl_gene_id, go$go_id)

In gs, there are a list of vectors where each vector can be thought as a gene set for the corresponding GO term.

Step 2. Complete genes in GO gene sets

GO has a tree structure where a parent term includes all genes annotated to its child terms. For every GO terms in gs, we need to merge genes from all its child or offspring terms to form a complete gene set for this GO term.

The hierarchical structure of GO terms is stored in the GO.db package. In the following code, bp_terms contains a vector of GO terms only in the Biological Process ontology. The variable GOBPOFFSPRING is simply a list where each element vector contains all offspring terms (child and remote downstream terms) of a GO term.

library(GO.db)
## Warning: package 'AnnotationDbi' was built under R version 4.2.1
## Warning: package 'BiocGenerics' was built under R version 4.2.1
## Warning: package 'Biobase' was built under R version 4.2.1
## Warning: package 'IRanges' was built under R version 4.2.1
## Warning: package 'S4Vectors' was built under R version 4.2.2
bp_terms = GOID(GOTERM)[Ontology(GOTERM) == "BP"]
GOBPOFFSPRING = as.list(GOBPOFFSPRING)

Now it is quite easy to merge genes from offspring terms. Just note as the final step, empty GO gene sets should be removed.

gs2 = lapply(bp_terms, function(nm) {
  go_id = c(nm, GOBPOFFSPRING[[nm]]) # self + offspring
  unique(unlist(gs[go_id]))
})
names(gs2) = bp_terms
gs2 = gs2[sapply(gs2, length) > 0]

Session info

## R version 4.2.0 (2022-04-22)
## Platform: x86_64-apple-darwin17.0 (64-bit)
## Running under: macOS Big Sur/Monterey 10.16
## 
## Matrix products: default
## BLAS:   /Library/Frameworks/R.framework/Versions/4.2/Resources/lib/libRblas.0.dylib
## LAPACK: /Library/Frameworks/R.framework/Versions/4.2/Resources/lib/libRlapack.dylib
## 
## locale:
## [1] C/UTF-8/C/C/C/C
## 
## attached base packages:
## [1] stats4    stats     graphics  grDevices utils     datasets  methods  
## [8] base     
## 
## other attached packages:
## [1] GO.db_3.16.0         AnnotationDbi_1.60.0 IRanges_2.32.0      
## [4] S4Vectors_0.36.1     Biobase_2.58.0       BiocGenerics_0.44.0 
## [7] biomaRt_2.54.0       knitr_1.42          
## 
## loaded via a namespace (and not attached):
##  [1] Rcpp_1.0.10            prettyunits_1.1.1      png_0.1-8             
##  [4] Biostrings_2.66.0      assertthat_0.2.1       rprojroot_2.0.3       
##  [7] digest_0.6.31          utf8_1.2.3             BiocFileCache_2.6.0   
## [10] R6_2.5.1               GenomeInfoDb_1.34.9    RSQLite_2.2.20        
## [13] evaluate_0.20          httr_1.4.4             pillar_1.8.1          
## [16] zlibbioc_1.44.0        rlang_1.0.6            progress_1.2.2        
## [19] curl_5.0.0             jquerylib_0.1.4        blob_1.2.3            
## [22] rmarkdown_2.20         pkgdown_2.0.7          textshaping_0.3.6     
## [25] desc_1.4.2             stringr_1.5.0          RCurl_1.98-1.10       
## [28] bit_4.0.5              compiler_4.2.0         xfun_0.37             
## [31] pkgconfig_2.0.3        systemfonts_1.0.4      htmltools_0.5.4       
## [34] tidyselect_1.2.0       KEGGREST_1.38.0        tibble_3.1.8          
## [37] GenomeInfoDbData_1.2.9 XML_3.99-0.13          fansi_1.0.4           
## [40] withr_2.5.0            crayon_1.5.2           dplyr_1.1.0           
## [43] dbplyr_2.3.0           rappdirs_0.3.3         bitops_1.0-7          
## [46] jsonlite_1.8.4         lifecycle_1.0.3        DBI_1.1.3             
## [49] magrittr_2.0.3         cli_3.6.0              stringi_1.7.12        
## [52] cachem_1.0.6           XVector_0.38.0         fs_1.6.1              
## [55] xml2_1.3.3             bslib_0.4.2            filelock_1.0.2        
## [58] ellipsis_0.3.2         ragg_1.2.5             vctrs_0.5.2           
## [61] generics_0.1.3         tools_4.2.0            bit64_4.0.5           
## [64] glue_1.6.2             purrr_1.0.1            hms_1.1.2             
## [67] fastmap_1.1.0          yaml_2.3.7             memoise_2.0.1         
## [70] sass_0.4.5