geposan/R/preset.R

165 lines
5.2 KiB
R
Raw Permalink Normal View History

2021-11-03 14:17:39 +01:00
#' Create a new preset.
#'
#' A preset is used to specify which methods and inputs should be used for an
#' analysis. Note that the genes to process should normally include the
#' reference genes to be able to assess the results later. The genes will be
#' filtered based on how many species have data for them. Afterwards, species
#' that still have many missing genes will also be excluded. See the different
2023-10-14 11:33:19 +02:00
#' method functions for the available methods: [distance()], [variation()],
#' [clustering()], [adjacency()], [correlation()] and [random_forest()].
2021-11-03 14:17:39 +01:00
#'
#' @param reference_gene_ids IDs of reference genes to compare to.
#' @param methods List of methods to apply.
2021-11-03 14:17:39 +01:00
#' @param species_ids IDs of species to include.
#' @param gene_ids IDs of genes to screen.
#' @param species_requirement The proportion of species a gene has to have
#' orthologs in in order for the gene to qualify.
#' @param gene_requirement The proportion of genes that a species has to have
#' in order for the species to be included in the analysis.
2021-11-03 14:17:39 +01:00
#'
#' @return The preset to use with [analyze()].
#'
#' @export
preset <- function(reference_gene_ids,
methods = all_methods(),
species_ids = geposan::species$id,
gene_ids = geposan::genes$id,
species_requirement = 0.25,
gene_requirement = 0.5) {
if (is.null(reference_gene_ids) || length(reference_gene_ids) < 1) {
stop(paste0(
"There has to be at least one reference gene for the preset to be ",
"valid. Please note that some methods may require more reference ",
"genes. It is best to select as many reference genes as possible."
))
}
if (is.null(methods) || length(methods) < 1) {
stop("Please select at least one method")
}
if (is.null(species_ids) || length(species_ids) < 1) {
stop(paste0(
"Please include at least one species. Please note that it is advisable ",
"to include more species (at least ten) in order to get enough data ",
"points per gene."
))
}
if (is.null(gene_ids) || length(gene_ids) < 1) {
stop(paste0(
"Please include at least one gene. Please note, that it is recommended ",
"to include all genes in the analysis (the default)."
))
}
# Prefilter distances.
distances <- geposan::distances[
species %chin% species_ids & gene %chin% gene_ids
2022-05-26 12:42:19 +02:00
]
# Count included species per gene.
genes_n_species <- distances[, .(n_species = .N), by = "gene"]
# Filter out genes with less too few existing orthologs.
2022-05-26 12:42:19 +02:00
gene_ids_filtered <- genes_n_species[
n_species >= species_requirement * length(species_ids),
2022-05-26 12:42:19 +02:00
gene
]
# Count included genes per species.
2022-06-20 14:30:51 +02:00
species_n_genes <- distances[, .(n_genes = .N), by = "species"]
# Filter out species that have too few of the genes.
species_ids_filtered <- species_n_genes[
n_genes >= gene_requirement * length(gene_ids_filtered),
species
]
n_species_excluded <- length(species_ids) - length(species_ids_filtered)
if (length(species_ids_filtered) < 10 && n_species_excluded >= 1) {
warning(glue::glue(
"Please note that {n_species_excluded} species have been excluded ",
"because they do not share enough orthologs."
))
}
# Check number of data points per gene again:
genes_n_species <- distances[
species %chin% species_ids_filtered & gene %chin% gene_ids_filtered,
.(n_species = .N),
by = "gene"
]
if (genes_n_species[, min(n_species)] < 2) {
warning(paste0(
"The preset contains genes with data for less than two species. Please ",
"select more species to get valid results."
))
}
2022-05-26 12:42:19 +02:00
reference_gene_ids_excluded <- reference_gene_ids[
!reference_gene_ids %chin% gene_ids_filtered
]
2022-05-26 12:42:19 +02:00
if (length(reference_gene_ids_excluded > 0)) {
warning(paste0(
"The following reference gene IDs are excluded from the preset ",
"because they don't have enough data: ",
paste(reference_gene_ids_excluded, collapse = ", ")
))
}
2022-05-26 12:42:19 +02:00
reference_gene_ids_included <- reference_gene_ids[
reference_gene_ids %chin% gene_ids_filtered
]
2022-05-26 12:42:19 +02:00
if (length(reference_gene_ids_included) < 1) {
stop(paste0(
"There has to be at least one reference gene for the preset to be ",
"valid. Please note that some methods may require more reference ",
"genes."
))
}
2022-05-26 12:42:19 +02:00
# The included data gets sorted to be able to produce predictable hashes
# for the object later.
structure(
list(
reference_gene_ids = sort(reference_gene_ids_included),
methods = methods,
species_ids = sort(species_ids_filtered),
2022-05-26 12:42:19 +02:00
gene_ids = sort(gene_ids_filtered)
),
class = "geposan_preset"
)
2021-11-03 14:17:39 +01:00
}
#' S3 method to print a preset object.
#'
2021-11-06 13:22:57 +01:00
#' @param x The preset to print.
#' @param ... Other parameters.
#'
2021-11-03 14:17:39 +01:00
#' @seealso [preset()]
#'
#' @export
2021-11-06 13:22:57 +01:00
print.geposan_preset <- function(x, ...) {
2022-05-26 12:42:19 +02:00
cat(sprintf(
paste0(
"geposan preset:",
"\n Reference genes: %i",
"\n Included methods: %s",
"\n Number of species: %i",
"\n Number of genes: %i",
"\n"
),
length(x$reference_gene_ids),
paste(sapply(x$methods, function(m) m$id), collapse = ", "),
length(x$species_ids),
length(x$gene_ids)
))
2021-11-03 14:17:39 +01:00
2022-05-26 12:42:19 +02:00
invisible(x)
2021-11-03 14:17:39 +01:00
}