tests/testthat/helpers.R

#' Expect identical ignoring $call slot
#'
#' @param object an R object
#' @param expected an R object
#' @param ... other args passed to expect_identical
#' @keywords internal
#' @return Logical, TRUE if all elements expect $call are identical
.almost_identical <- function(object, expected, ...) {
    object$call <- expected$call <- NULL
    expect_identical(object, expected, ...)
}


#' Expect equal numeric after rounding
#'
#' @param numeric_value numeric, outcome of test
#' @param expected numeric, reference to check
#' @param digits integer, decimals to round
#' @return logical, if TRUE, values are almost equal
#' @keywords internal
#' @examples
#' .numerically_close(3.414, 3.14, digits =2)
.expect_numerically_close <- function(numeric_value, expected, digits = 2) {
    require(testthat)
    expect_equal(round(numeric_value, digits = digits), round(expected, digits = digits))
}

#' Prevent calls to cat() and print() from showing
#'
#' @param x an R object
#' @keywords internal
#' @return x, the inputted R object
.quiet <- function(x) { 
  sink(tempfile()) 
  on.exit(sink()) 
  invisible(force(x)) 
} 


#' From input X and Y dataframes, yields the smallest set of training and testing
#' samples to remain valid for any mixOmics method. Caters sample selection to
#' if method requires multiblock, multigroup or multilevel frameworks.
#' 
#' @param X X dataframe for any mixOmics method. Can be a list of multiple dataframes if multiblock
#' @param Y Y dataframe or factor vector for any mixOmics method
#' @param S study factor vector for multigroup frameworks
#' @param ML repreated measures vector for multilevel frameworks
#' @param n.tr number of training samples (per class if DA, per level if MULTILEVEL, per study if MULTIGROUP)
#' @param n.te number of testing samples (per class if DA, per level if MULTILEVEL, per study if MULTIGROUP)
#' @param seed controls the sample selection seed
#' @return list of X, Y, study and multilevel components split by training and testing samples
#' @keywords internal
.minimal_sample_subset <- function(X=NULL, 
                                   Y=NULL, 
                                   S=NULL, 
                                   ML=NULL,
                                   n.tr=2,
                                   n.te=1,
                                   seed=16) {
  set.seed(seed)
  
  DA = is.factor(Y) # logical gate for DA framework
  MULTIGROUP = !is.null(S) # logical gate for multigroup framework
  MULTILEVEL = !is.null(ML) # logical gate for multilevel framework
  MULTIBLOCK = !is.data.frame(X) && !is.matrix(X) # logical gate for multiblock framework
  
  tr <- c() # initialise indicies for training and testing samples
  te <- c()
  
  if (MULTILEVEL) { # any multilevel method
    
    n.indivs <- 3 # default number of repeated samples to consider
    
    #if(DA) { n.indivs <- length(unique(Y))-1 } # if DA, set specific quantity
    
    # only look at the first n.indiv samples were measured the maximum amount of times
    indivs <- unname(which(table(ML) == max(table(ML))))[1:n.indivs] 
    
    for (i in 1:length(indivs)) { # for each repeated sample ...
      s <- indivs[i]
      
      rel.sam <- which(ML==s) # determine the corresponding rows
      tr.sam <- sample(rel.sam, n.tr, F) # take n.tr of these for training (1:n.tr+(i-1))
      te.sam <- setdiff(rel.sam, tr.sam) # and take n.te of these for testing
      
      tr <- c(tr, tr.sam)
      te <- c(te, te.sam)
      
    }
  } 
  else if(DA) { # if the framework is DA ... 
    
    for(c in unique(Y)) { # for each class ...
      
      if (MULTIGROUP) { # MINT.(s)PLSDA
        for (s in unique(S)){ # for each study ...
          # determine the rows with that class and for that study
          rel.sam <- intersect(which(Y==c), which(S==s)) 
          tr <- c(tr, rel.sam[1:n.tr]) # take first n.tr samples for training
          # if that samples's class and study is not already present in testing, add it
          if (!(s %in% S[te] || c %in% Y[te])) {te <- c(te, rel.sam[(n.tr+1):(n.tr+n.te)]) }
        }
      } else { # (BLOCK).(s)PLSDA
        rows <- which(Y == c)
        tr <- c(tr, rows[1:n.tr+1])
        te <- c(te, rows[(n.tr+1):(n.tr+n.te)])
      }
      
    }
    
    if (MULTIGROUP) { # ensure that all studies in training are present in testing
      tr.te.study.diff <- setdiff(unique(S[tr]), unique(S[te]))
      if (length(tr.te.study.diff) != 0) {
        for (s in tr.te.study.diff) {
          te <- c(te, which(S == s)[1])
        }
      }
    }
    
  } 
  else { 
    if (MULTIGROUP) { # MINT.(S)PLS
      for (s in unique(S)){
        rel.sam <- which(S==s)
        tr <- c(tr, rel.sam[1:n.tr])
        te <- c(te, rel.sam[(n.tr+1):(n.tr+n.te)])
      }
    } else { # (BLOCK).(s)PLS
      tr <- 1:n.tr
      te <- (n.tr+1):(n.tr+n.te)
    }
  }
  
  
  
  if(MULTIBLOCK) { # subset each block iteratively if multiblock
    X.tr <- list()         
    X.te <- list()
    
    for (block in names(X)) {
      X.tr[[block]] <- X[[block]][tr,]
      X.te[[block]] <- X[[block]][te,]
    }
  } else { # otherwise just subset X
    X.tr <- X[tr, ]
    X.te <- X[te, ]
  }
  
  if (DA) { # if Y is a factor, index list
    Y.tr <- Y[tr]
    Y.te <- Y[te]
  } else { # if Y is a data.frame, index df
    Y.tr <- Y[tr,]
    Y.te <- Y[te,]
  }
  
  out <- list(X.tr = X.tr,
              X.te = X.te,
              Y.tr = Y.tr,
              Y.te = Y.te)
  
  if (MULTILEVEL) { # include repeated measures
    out$ML.tr <- ML[tr]
    out$ML.te <- ML[te]
  }
  
  if (MULTIGROUP) { # include study
    out$S.tr <- as.factor(S[tr])
    out$S.te <- as.factor(S[te])
  }
  
  return(out)
}
mixOmicsTeam/mixOmics documentation built on Oct. 26, 2023, 6:48 a.m.