A fatty liver study on Mus musculus

Introduction

This vignette shows the utility of the FELLA package, which is based in a statistically normalised diffusion process (Picart-Armada et al. 2017), on non-human organisms. In particular, we will work on a multi-omic Mus musculus study. The original study (Gogiashvili et al. 2017) presents a mouse model of the non-alcoholic fatty liver disease (NAFLD). Metabolites in liver tissue from leptin-deficient ob/ob mice and wild type mice were compared using Nuclear Magnetic Resonance (NMR). Afterwards, quantitative real-time polymerase chain reaction (qRT-PCR) helped identify changes at the gene expression level. Finally, biological mechanisms behind NAFLD were elucidated by leveraging the data from both omics.

Building the database

The first step is to build the FELLA.DATA object for the mmu organism from the KEGG database (Kanehisa et al. 2016).

library(FELLA)
library(org.Mm.eg.db)
library(KEGGREST)

library(igraph)
library(magrittr)

set.seed(1)
# Filter overview pathways
graph <- buildGraphFromKEGGREST(
    organism = "mmu", 
    filter.path = c("01100", "01200", "01210", "01212", "01230"))

tmpdir <- paste0(tempdir(), "/my_database")
# Mke sure the database does not exist from a former vignette build
# Otherwise the vignette will rise an error 
# because FELLA will not overwrite an existing database
unlink(tmpdir, recursive = TRUE)  
buildDataFromGraph(
    keggdata.graph = graph, 
    databaseDir = tmpdir, 
    internalDir = FALSE, 
    matrices = "none", 
    normality = "diffusion", 
    niter = 100)

We load the FELLA.DATA object and two mappings (from gene symbol to entrez identifiers, and from enzyme EC numbers to their annotated entrez genes).

alias2entrez <- as.list(org.Mm.eg.db::org.Mm.egSYMBOL2EG)
entrez2ec <- KEGGREST::keggLink("enzyme", "mmu")
entrez2path <- KEGGREST::keggLink("pathway", "mmu")

fella.data <- loadKEGGdata(
    databaseDir = tmpdir, 
    internalDir = FALSE, 
    loadMatrix = "none"
)

Summary of the database:

fella.data
## General data:
## - KEGG graph:
##   * Nodes:  12095 
##   * Edges:  38248 
##   * Density:  0.0002614766 
##   * Categories:
##     + pathway [346]
##     + module [200]
##     + enzyme [1196]
##     + reaction [5975]
##     + compound [4378]
##   * Size:  5.9 Mb 
## - KEGG names are ready.
## -----------------------------
## Hypergeometric test:
## - Matrix not loaded.
## -----------------------------
## Heat diffusion:
## - Matrix not loaded.
## - RowSums are ready.
## -----------------------------
## PageRank:
## - Matrix not loaded.
## - RowSums not loaded.

In addition, we will store the ids of all the metabolites, reactions and enzymes in the database:

id.cpd <- getCom(fella.data, level = 5, format = "id") %>% names
id.rx <- getCom(fella.data, level = 4, format = "id") %>% names
id.ec <- getCom(fella.data, level = 3, format = "id") %>% names

Note on reproducibility

We want to emphasise that FELLA builds its FELLA.DATA object using the most recent version of the KEGG database. KEGG is frequently updated and therefore small changes can take place in the knowledge graph between different releases. The discussion on our findings was written at the date specified in the vignette header and using the KEGG release in the Reproducibility section.

Enrichment analysis

Defining the input and running the enrichment

Table 2 from the main body in (Gogiashvili et al. 2017) contains six metabolites that show significant changes between the experimental classes by a univariate test followed by multiple test correction. These are the start of our enrichment analysis:

cpd.nafld <- c(
    "C00020", # AMP
    "C00719", # Betaine
    "C00114", # Choline
    "C00037", # Glycine
    "C00160", # Glycolate
    "C01104"  # Trimethylamine-N-oxide
)

analysis.nafld <- enrich(
    compounds = cpd.nafld, 
    data = fella.data, 
    method = "diffusion", 
    approx = "normality")
## No background compounds specified. Default background will be used.
## Running diffusion...
## Computing p-scores through the specified distribution.
## Done.

Five compounds are successfully mapped to the graph object:

analysis.nafld %>% 
    getInput %>% 
    getName(data = fella.data)
## $C00020
## [1] "AMP"                        "Adenosine 5'-monophosphate"
## [3] "Adenylic acid"              "Adenylate"                 
## [5] "5'-AMP"                     "5'-Adenylic acid"          
## [7] "5'-Adenosine monophosphate" "Adenosine 5'-phosphate"    
## 
## $C00719
## [1] "Betaine"                 "Trimethylaminoacetate"  
## [3] "Glycine betaine"         "N,N,N-Trimethylglycine" 
## [5] "Trimethylammonioacetate"
## 
## $C00114
## [1] "Choline"     "Bilineurine"
## 
## $C00037
## [1] "Glycine"          "Aminoacetic acid" "Gly"             
## 
## $C00160
## [1] "Glycolate"          "Glycolic acid"      "Hydroxyacetic acid"
## 
## $C01104
## [1] "Trimethylamine N-oxide" "(CH3)3NO"

Likewise, one compound does not map:

getExcluded(analysis.nafld) 
## character(0)

The highlighted subgraph with the default parameters has the following appeareance, with large connected components that involve the metabolites in the input:

plot(
    analysis.nafld, 
    method = "diffusion", 
    data = fella.data, 
    nlimit = 250,  
    plotLegend = FALSE)

We will also extract all the p-scores and the suggested sub-network for further analysis:

g.nafld <-  generateResultsGraph(
    object = analysis.nafld, 
    data = fella.data, 
    method = "diffusion")

pscores.nafld <- getPscores(
    object = analysis.nafld, 
    method = "diffusion")

Examining the metabolites

From Table 2

The authors find 5 extra metabolites in Table 2 that are significant at p < 0.05 but do not appear after thresholding the false discovery rate at 5%. Such metabolites, highlighted in italics but without an asterisk, are also relevant and play a role in their discussion. We will examine how FELLA prioritises such metabolites:

cpd.nafld.suggestive <- c(
    "C00008", # ADP
    "C00791", # Creatinine
    "C00025", # Glutamate
    "C01026", # N,N-dimethylglycine
    "C00079", # Phenylalanine
    "C00299"  # Uridine
)
getName(cpd.nafld.suggestive, data = fella.data)
## $C00008
## [1] "ADP"                      "Adenosine 5'-diphosphate"
## 
## $C00791
## [1] "Creatinine"             "1-Methylglycocyamidine"
## 
## $C00025
## [1] "L-Glutamate"       "L-Glutamic acid"   "L-Glutaminic acid"
## [4] "Glutamate"        
## 
## $C01026
## [1] "N,N-Dimethylglycine" "Dimethylglycine"    
## 
## $C00079
## [1] "L-Phenylalanine"                          
## [2] "(S)-alpha-Amino-beta-phenylpropionic acid"
## 
## $C00299
## [1] "Uridine"

When checking if any of these metabolites are found in the reported sub-network, we find that C01026 is already reported:

V(g.nafld)$name %>% 
    intersect(cpd.nafld.suggestive) %>% 
    getName(data = fella.data)
## $C01026
## [1] "N,N-Dimethylglycine" "Dimethylglycine"

Abbreviated as DMG in their study, N,N-Dimethylglycine is a cornerstone of their findings. It is reported in Figure 6a as part of the folate-independent remethylation to explain the metabolic changes observed in the ob/ob mice. DMG is also mentioned in the conclusions as part of one of the most prominent alterations found in the study: a reduced conversion of betaine to DMG.

From Figure 6a

Figure 6a contains the metabolic context of the observed alterations, with processes such as transsulfuration and folate-dependent remethylation. These were identified with the help of gene expression analysis. We will now check for coincidences between the metabolites in Figure 6a, excluding choline and betaine for being in the input and DMG since it was already discussed.

cpd.new.fig6 <- c(
    "C00101", # THF
    "C00440", # 5-CH3-THF
    "C00143", # 5,10-CH3-THF
    "C00073", # Methionine
    "C00019", # SAM
    "C00021", # SAH
    "C00155", # Homocysteine
    "C02291", # Cystathione
    "C00097"  # Cysteine
)
getName(cpd.new.fig6, data = fella.data)
## $C00101
## [1] "Tetrahydrofolate"          "5,6,7,8-Tetrahydrofolate" 
## [3] "Tetrahydrofolic acid"      "THF"                      
## [5] "(6S)-Tetrahydrofolate"     "(6S)-Tetrahydrofolic acid"
## [7] "(6S)-THFA"                
## 
## $C00440
## [1] "5-Methyltetrahydrofolate"
## 
## $C00143
## [1] "5,10-Methylenetetrahydrofolate"      "(6R)-5,10-Methylenetetrahydrofolate"
## [3] "5,10-Methylene-THF"                 
## 
## $C00073
## [1] "L-Methionine"                      "Methionine"                       
## [3] "L-2-Amino-4methylthiobutyric acid"
## 
## $C00019
## [1] "S-Adenosyl-L-methionine" "S-Adenosylmethionine"   
## [3] "AdoMet"                  "SAM"                    
## 
## $C00021
## [1] "S-Adenosyl-L-homocysteine" "S-Adenosylhomocysteine"   
## 
## $C00155
## [1] "L-Homocysteine"                   "L-2-Amino-4-mercaptobutyric acid"
## [3] "Homocysteine"                    
## 
## $C02291
## [1] "L-Cystathionine"
## 
## $C00097
## [1] "L-Cysteine"                         "L-2-Amino-3-mercaptopropionic acid"

This time, there are no coincidences with the reported sub-network:

cpd.new.fig6 %in% V(g.nafld)$name
## [1] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE

However, we can further inquire whether the p-scores of such metabolites tend to be low among all the metabolites in the whole network from the fella.data object.

wilcox.test(
    x = pscores.nafld[cpd.new.fig6], # metabolites from fig6
    y = pscores.nafld[setdiff(id.cpd, cpd.new.fig6)], # rest of metabolites
    alternative = "less")
## 
##  Wilcoxon rank sum test with continuity correction
## 
## data:  pscores.nafld[cpd.new.fig6] and pscores.nafld[setdiff(id.cpd, cpd.new.fig6)]
## W = 9065, p-value = 0.002579
## alternative hypothesis: true location shift is less than 0

The test is indeed significant – despite FELLA does not directly report such metabolites, its metabolite ranking supports the claims by the authors.

Examining the genes

Cbs

The authors complement the metabolomic profilings with a differential gene expression study. One of the main findings is a change of Cbs expression levels. To link Cbs to the enrichment from FELLA, we will first map it to its EC number, 4.2.1.22 at the time of writing:

ec.cbs <- entrez2ec[[paste0("mmu:", alias2entrez[["Cbs"]])]] %>% 
    gsub(pattern = "ec:", replacement = "")

getName(fella.data, ec.cbs)
## $`4.2.1.22`
## [1] "cystathionine beta-synthase"               
## [2] "serine sulfhydrase"                        
## [3] "beta-thionase"                             
## [4] "methylcysteine synthase"                   
## [5] "cysteine synthase (incorrect)"             
## [6] "serine sulfhydrylase"                      
## [7] "L-serine hydro-lyase (adding homocysteine)"

In Figure 6a, the reaction linked to Cbs and catalysed by the enzyme 4.2.1.22 has the KEGG identifier R01290.

rx.cbs <- "R01290"

getName(fella.data, rx.cbs)
## $R01290
## [1] "L-serine hydro-lyase (adding homocysteine"          
## [2] "L-cystathionine-forming)"                           
## [3] "L-Serine + L-Homocysteine <=> L-Cystathionine + H2O"

As shown in Figure 6a, Cbs is not directly linked to the metabolites found through NMR, and nor the reaction neither the enzyme are suggested by FELLA:

c(rx.cbs, ec.cbs) %in% V(g.nafld)$name
## [1] FALSE FALSE

However, both of them have a relatively low p-score in their respective categories. This can be seen through the proportion of enzymes (resp. reactions) that show a p-score as low or lower than 4.2.1.22 (resp. R01290)

# enzyme
pscores.nafld[ec.cbs]
##  4.2.1.22 
## 0.3976106
mean(pscores.nafld[id.ec] <= pscores.nafld[ec.cbs])
## [1] 0.1697324
# reaction
pscores.nafld[rx.cbs]
##    R01290 
## 0.2750296
mean(pscores.nafld[id.rx] <= pscores.nafld[rx.cbs])
## [1] 0.03330544

It’s not surprising that none of them is directly reported, because none of the metabolites participating in the reaction is found in the input. The main evidence for finding Cbs is gene expression, and our approach gives indirect hints of this connection.

Bhmt

The alteration of Bhmt activity is related to the downregulation of Cbs. Despite not finding evidence of change in Bhmt expression, the authors argue that its inhibition would explain the increased betaine-to-DMG ratio in ob/ob mice. Such claim is also backed up by prior studies. To find out the role of Cbs in our analysis, we will again map it to its EC number, 2.1.1.5:

ec.bhmt <- entrez2ec[[paste0("mmu:", alias2entrez[["Bhmt"]])]] %>% 
    gsub(pattern = "ec:", replacement = "")

getName(fella.data, ec.bhmt)
## $`2.1.1.5`
## [1] "betaine---homocysteine S-methyltransferase"
## [2] "betaine-homocysteine methyltransferase"    
## [3] "betaine-homocysteine transmethylase"

This time, FELLA not only reports it, but also its associated reaction R02821 (represented by an arrow in Figure 6a) and both of its metabolites. While betaine was already an input metabolite, DMG was a novel finding as discussed earlier

ec.bhmt %in% V(g.nafld)$name
## [1] TRUE
"R02821" %in% V(g.nafld)$name
## [1] TRUE

This illustrates how FELLA can translate knowledge from dysregulated metabolites to other molecular levels, such as reactions and enzymes.

Slc22a5

The decrease of Bhmt activity is later connected to the upregulation of Slc22a5, also proved within the original study. However, Slc22a5 does not map to any EC number and therefore it cannot be found through FELLA:

entrez.slc22a5 <- alias2entrez[["Slc22a5"]]
entrez.slc22a5 %in% names(entrez2ec)
## [1] FALSE

As a matter of fact, the only connection that can be found from KEGG is the role of Slc22a5 in the Choline metabolism in cancer pathway.

path.slc22a5 <- entrez2path[paste0("mmu:", entrez.slc22a5)] %>% 
    gsub(pattern = "path:", replacement = "")

getName(fella.data, path.slc22a5)
## $mmu05231
## [1] "Choline metabolism in cancer - Mus musculus (house mouse)"

Coincidentally, this pathway is reported in the sub-graph:

path.slc22a5 %in% V(g.nafld)$name
## [1] TRUE

Genes from Figure 3

We also examined if genes from Table 3 were reachable in our analysis. These five literature-derived genes were experimentally confirmed to show gene expression changes, in order to prove that RNA extracted after the metabolomic profiling was still reliable for further transcriptomic analyses. However, only Scd2 maps to an enzymatic family:

symbol.fig3 <- c(
    "Cd36",
    "Scd2", 
    "Apoa4", 
    "Lcn2", 
    "Apom")

entrez.fig3 <- alias2entrez[symbol.fig3] %>% unlist %>% unique
ec.fig3 <- entrez2ec[paste0("mmu:", entrez.fig3)] %T>% 
    print %>%
    unlist %>% 
    unique %>% 
    na.omit %>% 
    gsub(pattern = "ec:", replacement = "")
##           <NA>      mmu:20250           <NA>           <NA>           <NA> 
##             NA "ec:1.14.19.1"             NA             NA             NA
getName(fella.data, ec.fig3)
## $`1.14.19.1`
## [1] "stearoyl-CoA 9-desaturase"                         
## [2] "Delta9-desaturase"                                 
## [3] "acyl-CoA desaturase"                               
## [4] "fatty acid desaturase"                             
## [5] "stearoyl-CoA, hydrogen-donor:oxygen oxidoreductase"

Such family is not reported in our sub-graph

ec.fig3 %in% V(g.nafld)$name
## [1] FALSE

In addition, its p-score is high compared to other enzymes

pscores.nafld[ec.fig3]
## 1.14.19.1 
## 0.5735912
mean(pscores.nafld[id.ec] <= pscores.nafld[ec.fig3])
## [1] 0.7382943

The fact that only one gene mapped to an EC number hinders the potential findings using FELLA, and is probably the main reason why FELLA missed Scd2. In addition, FELLA defines a knowledge model that offers simplicity and interpretability, at the cost of introducing limitations on how sophisticated its findings can be.

Genes from Table S2

In parallel with the original study, and cited within its main body, gene array expression data was collected (Godoy et al. 2016) and its hits are included in the supplementary Table S2 from (Gogiashvili et al. 2017). These genes include the already discussed Cbs. We will attempt to link the genes marked as significantly changed to our reported sub-network. In contrast with Figure 3, all the genes map to an EC number:

symbol.tableS2 <- c(
    "Mat1a",
    "Ahcyl2", 
    "Cbs",
    "Mat2b",  
    "Mtr")
entrez.tableS2 <- alias2entrez[symbol.tableS2] %>% unlist %>% unique
ec.tableS2 <- entrez2ec[paste0("mmu:", entrez.tableS2)] %T>%
    print %>%
    unlist %>% 
    unique %>% 
    na.omit %>% 
    gsub(pattern = "ec:", replacement = "")
##     mmu:11720     mmu:74340     mmu:12411    mmu:108645    mmu:238505 
##  "ec:2.5.1.6" "ec:3.13.2.1" "ec:4.2.1.22"  "ec:2.5.1.6" "ec:2.1.1.13"

None of these EC families are reported in the sub-graph:

ec.tableS2 %in% V(g.nafld)$name
## [1] FALSE FALSE FALSE FALSE

But in this case, their scores tend to be lower than the rest of enzymes:

wilcox.test(
    x = pscores.nafld[ec.tableS2], # enzymes from table S2
    y = pscores.nafld[setdiff(id.ec, ec.tableS2)], # rest of enzymes
    alternative = "less")
## 
##  Wilcoxon rank sum test with continuity correction
## 
## data:  pscores.nafld[ec.tableS2] and pscores.nafld[setdiff(id.ec, ec.tableS2)]
## W = 814, p-value = 0.01143
## alternative hypothesis: true location shift is less than 0

These findings suggest that if the annotation database is complete enough, FELLA can provide a meaningful priorisisation of the enzymes surrounding the affected metabolites.

Conclusions

FELLA has been used to give a biological meaning to a list of 6 metabolites extracted from a multi-omic study of a mouse model of NAFLD. It has been able to reproduce some findings at the metabolite and gene expression levels, whereas most of the times missed entities would still present a low ranking compared to their background in the database.
The bottom line from our analysis in the present vignette is that FELLA not only works on human studies, but also generalises to animal models.

Reproducibility

This is the result of running sessionInfo()

sessionInfo()
## R version 4.4.2 (2024-10-31)
## Platform: x86_64-pc-linux-gnu
## Running under: Ubuntu 24.04.1 LTS
## 
## Matrix products: default
## BLAS:   /usr/lib/x86_64-linux-gnu/openblas-pthread/libblas.so.3 
## LAPACK: /usr/lib/x86_64-linux-gnu/openblas-pthread/libopenblasp-r0.3.26.so;  LAPACK version 3.12.0
## 
## locale:
##  [1] LC_CTYPE=en_US.UTF-8       LC_NUMERIC=C              
##  [3] LC_TIME=en_US.UTF-8        LC_COLLATE=C              
##  [5] LC_MONETARY=en_US.UTF-8    LC_MESSAGES=en_US.UTF-8   
##  [7] LC_PAPER=en_US.UTF-8       LC_NAME=C                 
##  [9] LC_ADDRESS=C               LC_TELEPHONE=C            
## [11] LC_MEASUREMENT=en_US.UTF-8 LC_IDENTIFICATION=C       
## 
## time zone: Etc/UTC
## tzcode source: system (glibc)
## 
## attached base packages:
## [1] stats4    stats     graphics  grDevices utils     datasets  methods  
## [8] base     
## 
## other attached packages:
##  [1] magrittr_2.0.3       igraph_2.1.3         KEGGREST_1.47.0     
##  [4] org.Mm.eg.db_3.20.0  AnnotationDbi_1.69.0 IRanges_2.41.2      
##  [7] S4Vectors_0.45.2     Biobase_2.67.0       BiocGenerics_0.53.3 
## [10] generics_0.1.3       FELLA_1.27.0         BiocStyle_2.35.0    
## 
## loaded via a namespace (and not attached):
##  [1] sass_0.4.9              RSQLite_2.3.9           lattice_0.22-6         
##  [4] digest_0.6.37           evaluate_1.0.3          grid_4.4.2             
##  [7] fastmap_1.2.0           blob_1.2.4              plyr_1.8.9             
## [10] jsonlite_1.8.9          Matrix_1.7-1            GenomeInfoDb_1.43.2    
## [13] DBI_1.2.3               BiocManager_1.30.25     httr_1.4.7             
## [16] UCSC.utils_1.3.1        Biostrings_2.75.3       jquerylib_0.1.4        
## [19] cli_3.6.3               rlang_1.1.5             crayon_1.5.3           
## [22] XVector_0.47.2          bit64_4.6.0-1           cachem_1.1.0           
## [25] yaml_2.3.10             tools_4.4.2             memoise_2.0.1          
## [28] GenomeInfoDbData_1.2.13 curl_6.1.0              buildtools_1.0.0       
## [31] vctrs_0.6.5             R6_2.5.1                png_0.1-8              
## [34] lifecycle_1.0.4         bit_4.5.0.1             pkgconfig_2.0.3        
## [37] bslib_0.8.0             glue_1.8.0              Rcpp_1.0.14            
## [40] xfun_0.50               sys_3.4.3               knitr_1.49             
## [43] htmltools_0.5.8.1       rmarkdown_2.29          maketools_1.3.1        
## [46] compiler_4.4.2

KEGG version:

cat(getInfo(fella.data))
## T01002           Mus musculus (house mouse) KEGG Genes Database
## mmu              Release 113.0+/01-19, Jan 25
##                  Kanehisa Laboratories
##                  26,040 entries
## 
## linked db        pathway
##                  brite
##                  module
##                  ko
##                  mgi
##                  genome
##                  enzyme
##                  ncbi-geneid
##                  ncbi-proteinid
##                  uniprot

Date of generation:

date()
## [1] "Sun Jan 19 06:09:32 2025"

Image of the workspace (for submission):

tempfile(pattern = "vignette_mmu_", fileext = ".RData") %T>% 
    message("Saving workspace to ", .) %>% 
    save.image(compress = "xz")
## Saving workspace to /tmp/RtmpCgh1a5/vignette_mmu_173c1ae730b9.RData

References

Godoy, Patricio, Agata Widera, Wolfgang Schmidt-Heck, Gisela Campos, Christoph Meyer, Cristina Cadenas, Raymond Reif, et al. 2016. “Gene Network Activity in Cultivated Primary Hepatocytes Is Highly Similar to Diseased Mammalian Liver Tissue.” Archives of Toxicology 90 (10): 2513–29.
Gogiashvili, Mikheil, Karolina Edlund, Kathrin Gianmoena, Rosemarie Marchan, Alexander Brik, Jan T Andersson, Jörg Lambert, et al. 2017. “Metabolic Profiling of Ob/Ob Mouse Fatty Liver Using HR-MAS 1 h-NMR Combined with Gene Expression Analysis Reveals Alterations in Betaine Metabolism and the Transsulfuration Pathway.” Analytical and Bioanalytical Chemistry 409 (6): 1591–1606.
Kanehisa, Minoru, Miho Furumichi, Mao Tanabe, Yoko Sato, and Kanae Morishima. 2016. “KEGG: New Perspectives on Genomes, Pathways, Diseases and Drugs.” Nucleic Acids Research 45 (D1): D353–61.
Picart-Armada, Sergio, Francesc Fernández-Albert, Maria Vinaixa, Miguel A Rodríguez, Suvi Aivio, Travis H Stracker, Oscar Yanes, and Alexandre Perera-Lluna. 2017. “Null Diffusion-Based Enrichment for Metabolomics Data.” PloS One 12 (12): e0189012.