#' Converts bed style data.frame to GRanges
#'
#' For info on columns, see:
#' https://www.ensembl.org/info/website/upload/bed.html
#' @param x A \code{\link{data.frame}} from imported bed-file,
#' to convert to GRanges
#' @param skip.name default (TRUE), skip name column (column 4)
#' @return a \code{\link{GRanges}} object from bed
#' @family utils
#' @keywords internal
bedToGR <- function(x, skip.name = TRUE) {
if (ncol(x) < 3) stop("bed file must have a least 3 columns!")
starts <- x[, 2L] + 1L
ends <- x[, 3L]
gr <- GRanges(x[, 1L], IRanges(starts, ends))
bed4 <- (ncol(x) >= 4) & !skip.name #name column
if (bed4) mcols(gr) <- S4Vectors::DataFrame(mcols(gr), name = x[, 4L])
bed5 <- ncol(x) >= 5 # score column
if (bed5) mcols(gr) <- S4Vectors::DataFrame(mcols(gr), score = x[, 5L])
bed6 <- ncol(x) >= 6
if (bed6) strand(gr) <- x[, 6L]
if (ncol(x) > 6L) mcols(gr) <- x[, 7L:ncol(x)]
return(gr)
}
#' Internal GRanges loader from fst data.frame
#' @param df a data.frame with columns minimum 4 columns:
#' seqnames, start, strand\cr
#' Additional specific columns are:\cr
#' - width (if not set, width is set to 1 for all reads)\cr
#' Additional columns will be assigned as meta columns
#' @inheritParams import.ofst
#' @return GRanges object
#' @importFrom S4Vectors new2
#' @keywords internal
getGRanges <- function(df, seqinfo = NULL) {
if (!all(c("seqnames", "start", "strand") %in% colnames(df)))
stop("df must at minimum have 4 columns named: seqnames, start, width and strand")
widths <- if ("width" %in% colnames(df)) {
df$width
} else rep.int(1L, nrow(df))
ranges <- new2("IRanges", start = df$start,
width = widths,
NAMES = df$NAMES,
elementMetadata = NULL,
check = FALSE)
if (is.null(levels(df$seqnames))) {
df$seqnames <- factor(df$seqnames, levels = unique(df$seqnames))
}
if (!is.null(seqinfo)) {
stopifnot(is(seqinfo, "Seqinfo"))
} else seqinfo <- Seqinfo(levels(df$seqnames))
df$NAMES <- NULL
mcols_hits <- !(colnames(df) %in% c("seqnames", "start", "strand", "width"))
has_mcols <- any(mcols_hits)
if (!has_mcols){
mcols <- NULL
} else {
mcols <- df[,which(mcols_hits)]
if (sum(mcols_hits) == 1) {
mcols <- data.frame(mcols)
names(mcols) <- names(df)[mcols_hits]
}
}
mcols <- S4Vectors:::normarg_mcols(mcols, "GRanges", nrow(df))
new2("GRanges", seqnames = Rle(df$seqnames), ranges = ranges, strand = Rle(df$strand),
elementMetadata = mcols, seqinfo = seqinfo, check = FALSE)
}
#' Internal GAlignments loader from fst data.frame
#' @param df a data.frame with columns minimum 4 columns:
#' seqnames, start ("pos" in final GA object), cigar and strand.\cr
#' Additional columns will be assigned as meta columns
#' @inheritParams import.ofst
#' @return GAlignments object
#' @importFrom S4Vectors new2
#' @keywords internal
getGAlignments <- function(df, seqinfo = NULL) {
if (!all(c("seqnames", "start", "cigar", "strand") %in% colnames(df)))
stop("df must at minimum have 4 columns named: seqnames, start, cigar and strand")
if (nrow(df) == 0) return(GenomicAlignments::GAlignments())
if (is.null(levels(df$seqnames))) {
df$seqnames <- factor(df$seqnames, levels = unique(df$seqnames))
}
if (!is.null(seqinfo)) {
stopifnot(is(seqinfo, "Seqinfo"))
} else seqinfo <- Seqinfo(levels(df$seqnames))
names <- df$NAMES
if (!is.null(df$NAMES)) df$NAMES <- NULL
if (ncol(df) == 4){
mcols <- NULL
} else {
mcols <- df[,5:ncol(df)]
if (ncol(df) == 5) { # Hm... Is this safe ? What if a score is there ?
mcols <- data.frame(mcols)
names(mcols) <- names(df)[5]
}
}
df$strand <- factor(df$strand, levels = c("+", "-", "*"))
mcols <- S4Vectors:::normarg_mcols(mcols, "GRanges", nrow(df))
new2("GAlignments", NAMES = names, seqnames = Rle(df$seqnames), start = df$start,
cigar = as.character(df$cigar), strand = Rle(df$strand), elementMetadata = mcols,
seqinfo = seqinfo, check = FALSE)
}
#' Internal GAlignmentPairs loader from fst data.frame
#' @param df a data.frame with columns minimum 6 columns:
#' seqnames, start1/start2 (integers), cigar1/cigar2 and strand\cr
#' Additional columns will be assigned as meta columns
#' @inheritParams import.ofst
#' @return GAlignmentPairs object
#' @importFrom S4Vectors new2
#' @keywords internal
getGAlignmentsPairs <- function(df, strandMode = 0, seqinfo = NULL) {
if (nrow(df) == 0) {
return(GenomicAlignments::GAlignmentPairs(first = GAlignments(),
last = GAlignments(),
strandMode = strandMode))
}
if (is.null(levels(df$seqnames))) {
df$seqnames <- factor(df$seqnames, levels = unique(df$seqnames))
}
if (!is.null(seqinfo)) {
stopifnot(is(seqinfo, "Seqinfo"))
} else seqinfo <- Seqinfo(levels(df$seqnames))
names <- df$NAMES
if (!is.null(df$NAMES)) df$NAMES <- NULL
if (ncol(df) == 6){
mcols <- NULL
} else {
mcols <- df[,7:ncol(df)]
if (ncol(df) == 7) { # Hm... Is this safe ? What if a score is there ?
mcols <- data.frame(mcols)
names(mcols) <- names(df)[7]
}
}
mcols <- S4Vectors:::normarg_mcols(mcols, "GRanges", nrow(df))
# reverse strand for last
levels = c("+", "-", "*")
strand2 <- strandTemp <- df$strand <-
factor(df$strand, levels = levels)
strandTemp[strand2 == "+"] <- "-"
strandTemp[strand2 == "-"] <- "+"
strand2 <- strandTemp
new2("GAlignmentPairs",
first = new2("GAlignments", NAMES = names, seqnames = Rle(df$seqnames), start = df$start1,
cigar = as.character(df$cigar1), strand = Rle(df$strand),
seqinfo = seqinfo, check = FALSE,
elementMetadata = DataFrame(data.frame(matrix(nrow = nrow(df), ncol = 0)))),
last = new2("GAlignments", NAMES = names, seqnames = Rle(df$seqnames), start = df$start2,
cigar = as.character(df$cigar2), strand = Rle(factor(strand2, levels)),
seqinfo = seqinfo, check = FALSE,
elementMetadata = DataFrame(data.frame(matrix(nrow = nrow(df), ncol = 0)))),
isProperPair = rep(TRUE, nrow(df)), strandMode = as.integer(strandMode),
elementMetadata = mcols, check = FALSE)
}
#' Find pair of forward and reverse strand wig / bed files and
#' paired end bam files split in two
#'
#' @param paths a character path at least one .wig / .bed file
#' @param f Default (c("forward", "fwd")
#' a character vector for forward direction regex.
#' @param r Default (c("reverse", "rev")
#' a character vector for reverse direction regex.
#' @param format default "wig", for bed do "bed". Also searches
#' compressions of these variants.
#' @return if not all are paired, return original list,
#' if they are all paired, return a data.table with matches as 2 columns
#' @keywords internal
findNGSPairs <- function(paths, f = c("forward", "fwd"),
r = c("reverse", "rev"), format = "wig") {
if (length(paths) == 0) return(paths)
f <- paste0(f, "\\.", format, "*", collapse = "|")
r <- paste0(r, "\\.", format, "*", collapse = "|")
forwardPath <- grep(f, paths)
reversePath <- grep(r, paths)
if ((length(forwardPath) != length(reversePath)) |
length(forwardPath) == 0 | length(reversePath) == 0) return(paths)
dt <- data.table(forward = paths[forwardPath],
reverse = paths[reversePath], match = FALSE)
dt.sub <- dt[, .(forward = gsub(pattern = f, x = forward, ""),
reverse = gsub(pattern = r, x = reverse, ""))]
matches <- chmatch(dt.sub$forward, dt.sub$reverse)
if (!anyNA(matches)) {
dt <- dt[, .(forward, reverse = reverse[matches], match = TRUE)]
}
if (all(dt$match)) {
return(dt)
}
return(paths)
}
#' Remove file extension of path
#'
#' Allows removal of compression
#' @param path character path (allows multiple paths)
#' @param basename relative path (TRUE) or full path (FALSE)? (default: FALSE)
#' @importFrom tools file_ext
#' @return character path without file extension
#' @keywords internal
remove.file_ext <- function(path, basename = FALSE) {
fext <- file_ext(path)
compressions <- c("gzip", "gz", "bgz", "zip")
areCompressed <- fext %in% compressions
regex <- paste0("*\\.",fext,"$")
if (any(areCompressed)) {
ext <- file_ext(file_path_sans_ext(path[areCompressed], compression = FALSE))
regex[areCompressed] <- paste0("*\\.",ext,"\\.", fext[areCompressed],"$")
}
out <- c()
for (p in seq_along(path)) {
new <- gsub(pattern = regex[p], "", path[p])
out <- c(out, new)
}
return(if(basename){basename(out)}else out)
}
#' A wrapper for seqlevelsStyle
#'
#' To make sure chromosome naming is correct (chr1 vs 1 vs I etc)
#' @param range a ranged object, (GRanges, GAlignment etc)
#' @param chrStyle a GRanges object, TxDb, FaFile,
#' , a \code{\link{seqlevelsStyle}} or \code{\link{Seqinfo}}.
#' (Default: NULL) to get seqlevelsStyle from. In addition if it
#' is a Seqinfo object, seqinfo will be updated.
#' Example of seqlevelsStyle update:
#' Is chromosome 1 called chr1 or 1,
#' is mitocondrial chromosome called MT or chrM etc.
#' Will use 1st seqlevel-style if more are present.
#' Like: c("NCBI", "UCSC") -> pick "NCBI"
#' @return a GAlignment/GRanges object depending on input.
#' @keywords internal
matchSeqStyle <- function(range, chrStyle = NULL) {
# TODO: Check if this makes it safer:
# if (tryCatch(seqlevelsStyle(cage) <- seqlevelsStyle(fiveUTRs),
# error = function(e) {TRUE}) == TRUE) {
# warning("seqlevels of CAGE/fiveUTRs are not standardized, check them.")
# } else {
# seqlevelsStyle(cage) <- seqlevelsStyle(fiveUTRs)
# }
if (!is.null(chrStyle)) {
valid_seq_style <- seqlevelsStyleSafe(range)
if (!is.null(valid_seq_style)) {
if (is.character(chrStyle)) {
seqlevelsStyle(range) <- chrStyle[1]
} else if (is.gr_or_grl(chrStyle) | is(chrStyle, "TxDb") |
is(chrStyle, "FaFile") | is(chrStyle, "Seqinfo")) {
style_is_different <- !any(seqlevelsStyle(range) %in% seqlevelsStyle(chrStyle)[1])
if (style_is_different) {
seqlevelsStyle(range) <- seqlevelsStyle(chrStyle)[1]
}
} else stop("chrStyle must be valid GRanges object,",
"or a valid chr style!")
}
}
if (is(chrStyle, "Seqinfo")) {
seqlevels(range) <- seqlevels(chrStyle)
seqinfo(range) <- chrStyle
}
return(range)
}
seqlevelsStyleTest <- function(x) {
try(seqlevelsStyle(x), silent = TRUE)
}
seqlevelsStyleSafe <- function(x) {
res <- seqlevelsStyleTest(x)
if (!is(res, "try-error")) return(res)
return(NULL)
}
#' Find optimized subset of valid reads
#'
#' Keep only the ones that overlap within the grl ranges.
#' Also sort them in the end
#' @inheritParams validSeqlevels
#' @return the reads as GRanges, GAlignment or GAlignmentPairs
#' @importFrom GenomicAlignments first
#' @family utils
#' @keywords internal
optimizeReads <- function(grl, reads) {
if (is(reads, "covRle")) return(reads)
seqMatch <- validSeqlevels(grl, reads)
reads <- keepSeqlevels(reads, seqMatch, pruning.mode = "coarse")
reads <- reads[countOverlaps(reads, grl, type = "within") > 0]
reads <- if (is(reads, "GAlignmentPairs")) {
message("For GAlignmentPairs using 'first' read of pair only,",
" Make sure it is correct!")
reads <- reads[order(GenomicAlignments::first(reads))]
} else reads <- sort(reads)
if (length(reads) == 0) warning("No reads left in 'reads' after",
"optimisation!")
return(reads)
}
#' Transform object
#'
#' Similar to normal transform like log2 or log10.
#' But keep 0 values as 0, to avoid Inf values and negtive values
#' are made as -scale(abs(x)), to avoid NaN values.
#' @param x a numeric vector or data.frame/data.table of numeric columns
#' @param scale a function, default log2, which function to transform with.
#' @param by.reference logical, FALSE. if TRUE, update object by reference
#' if it is data.table.
#' @return same object class as x, with transformed values
#' @keywords internal
pseudo.transform <- function(x, scale = log2, by.reference = FALSE) {
if (!by.reference)
x <- copy(x)
xl <- x
if (is.numeric(x)) {
x[(x > 0)] <- scale(x[(x > 0)])
x[(xl < 0)] <- - scale(abs(x[(xl < 0)]))
} else if (is.data.frame(x)) {
for (i in names(x)) {
b <- x[, get(i)]
bigger <- b > 0
smaller <- (b < 0)
b[bigger] <- scale(b[bigger])
b[smaller] <- -scale(b[abs(smaller)])
x[, (i):= b]
}
} else stop("x must be numeric or data.frame/data.table/matrix")
return(x)
}
#' Create all unique combinations pairs possible
#'
#' Given a character vector, get all unique combinations of 2.
#' @param x a character vector, will unique elements for you.
#' @return a list of character vector pairs
#' @importFrom utils combn
#' @export
#' @examples
#' df <- ORFik.template.experiment()
#' ORFik:::combn.pairs(df[, "libtype"])
combn.pairs <- function(x) {
pairs <- list() # creating compairisons :list of pairs
comparisons.design <- unique(x)
my_comparison <- combn(unique(comparisons.design), 2)
pairs <- list()
for (i in seq(ncol(my_comparison))) {
pairs[[i]] <- c(my_comparison[1, i], my_comparison[2, i])
}
return(pairs)
}
#' A fast ftp directory check
#'
#' Check if ftp directory exists
#' @param url.dir character, url to a ftp directory.
#' @param report.error logical, FALSE. If TRUE,
#' stop and report error.
#' @return logical, TRUE if url directory exists
#' @importFrom httr http_error
#' @keywords internal
exists.ftp.dir.fast <- function(url.dir, report.error = FALSE) {
url.dir.not.exists <- tryCatch(
expr = {
httr::http_error(url.dir)
},
error = function(e){
e
})
if (is.logical(url.dir.not.exists)) { # Directory exists
return(TRUE)
} else { # Directory does not exist
if (report.error) {
msg <- url.dir.not.exists$message
if (length(grep("Timeout was reached.", msg)) == 1 |
length(grep("Recv failure:", msg)) == 1) {
stop("FTP port 21 is not connecting to || ", url.dir,
" || wait 5 minutes, if it still does not work",
" check your firewall and FTP settings!")
} else if (length(grep("Could not resolve host", msg)) == 1){
stop("Server is not up:,", url.dir, "check again in 5 minutes!")
} else if (length(grep("Server denied you to change", msg)) == 1){
stop("Directory does not exist: ", url.dir)
}
}
}
# Else return that file does not exist
return(FALSE)
}
#' A fast ftp file check
#'
#' Check if ftp file exists
#' @param url character, url to a ftp file
#' @param report.error logical, FALSE. If TRUE,
#' stop and report error.
#' @return logical, TRUE if file exists
#' @importFrom httr http_error
#' @keywords internal
exists.ftp.file.fast <- function(url, report.error = FALSE) {
# Stop early if directory does not exist
if (!exists.ftp.dir.fast(paste0(dirname(url), "/"), report.error))
return(FALSE)
# Else check if file exists
url.not.exists <- tryCatch(
expr = {
httr::http_error(url)
},
error = function(e){
e
})
if (is.logical(url.not.exists)) { # File exists
return(TRUE)
} else { # File does not exist
if (report.error) {
msg <- url.not.exists$message
if (length(grep("Timeout was reached.", msg)) == 1 |
length(grep("Recv failure:", msg)) == 1) {
stop("FTP port 21 is not connecting to || ", url,
" || wait 5 minutes, if it still does not work",
" check your internet connection, firewall and FTP settings!")
} else if (length(grep("Given file does not exist", msg)) == 1){
stop("File does not exist,", url)
}
}
}
return(FALSE)
}
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.