R/downscaleTrain.R

Defines functions downs.train downscaleTrain

Documented in downscaleTrain downs.train

##     downscaleTrain.R Calibration of downscaling methods
##
##     Copyright (C) 2018 Santander Meteorology Group (http://www.meteo.unican.es)
##
##     This program is free software: you can redistribute it and/or modify
##     it under the terms of the GNU General Public License as published by
##     the Free Software Foundation, either version 3 of the License, or
##     (at your option) any later version.
## 
##     This program is distributed in the hope that it will be useful,
##     but WITHOUT ANY WARRANTY; without even the implied warranty of
##     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
##     GNU General Public License for more details.
## 
##     You should have received a copy of the GNU General Public License
##     along with this program.  If not, see <http://www.gnu.org/licenses/>.

#' @title Calibration of downscaling methods
#' @description Calibration of downscaling methods. Currently analogs, generalized linear models (GLM) and Neural Networks (NN) are available. 
#' @param obj The object as returned by \code{\link[downscaleR]{prepareData}}.
#' @param method Character string indicating the type of method/transfer function. Currently accepted values are \code{"analogs"}, \code{"GLM"} or \code{"NN"}.
#' @param condition Inequality operator to be applied to the given \code{"threshold"}. Only the days that satisfy the condition will be used for training the model.
#' \code{"GT"} = greater than the value of \code{threshold}, \code{"GE"} = greater or equal,
#' \code{"LT"} = lower than, \code{"LE"} = lower or equal than.
#' @param threshold Numeric value. Threshold used as reference for the condition. Default is NULL. If a threshold value is supplied with no specificaction of the argument \code{condition}. Then condition is set to \code{"GE"}.
#' @param model.verbose A logic value. Indicates wether the information concerning the model infered is limited to the 
#' essential information (model.verbose = FALSE)  or a more detailed information (model.verbose = TRUE, DEFAULT). This is
#' recommended when you want to save memory. Only operates for GLM.
#' @param predict A logic value. Should the prediction on the training set should be returned? Default is TRUE.
#' @param simulate A logic value indicating whether we want to simulate or not based on the GLM distributional parameters when prediting on the train set. Only relevant when perdicting with a GLM. Default to FALSE. 
#' @param ... Optional parameters. These parameters are different depending on the method selected. Every parameter has a default value set in the atomic functions in case that no selection is wanted. 
#' Everything concerning these parameters is explained in the section \code{Details}. 
#' However, if wanted, the atomic functions can be seen here: \code{\link[downscaleR]{glm.train}} and \code{\link[deepnet]{nn.train}}.  

#' @details The function can downscale in both global and local mode, though not simultaneously. 
#' If there is perfect collinearity among predictors, then the matrix will not be invertible and the downscaling will fail.
#' We recommend to get rid of the NaN/NA values before calling the function.
#' 
#' \strong{Analogs}
#' The optional parameters of this method are:
#' \itemize{
#' \item \code{n.analogs} An integer. Number of analogs. Default is 4.
#' \item \code{sel.fun} A string. Select a function to apply to the analogs selected for a given observation. Options are 
#' "mean", "wmean" (i.e., weighted mean), "max", "min", "median", "prcXX" 
#' (i.e., prc85 means the 85th percentile of the analogs values distribution). Default is "mean".
#' the function applied to the analogs values, (i.e., sel.fun = c("mean","max","min","median","prcXX"), with default "mean") 
#' and the temporal window, (i.e., window = 0).
#' \item \code{window} An integer. Window of days removed when selecting analogs. 
#' If window = 7, then 7 days after the observation date and the 7 days before the observation date are removed. Default is 0.
#' \item \code{n.random} An integer. Choose N random analogs among the closest n.analogs. Default is NULL.
#' }
#' More information can be found in \code{\link[downscaleR]{analogs.train}}
#' 
#' \strong{Generalized Linear Models (GLM)}
#' The optional parameters depends on the \code{fitting} optional parameter:
#' \itemize{
  #' \item \code{fitting} A string indicating the types of objective functions and how to fit the linear model.
  #' \itemize{
    #' \item \code{fitting = NULL} In this case the generalized linear model uses the \code{\link[stats]{glm}} function to fit the linear model. 
    #' This is the default option.
    #' The optional parameters when fitting = NULL are:
    #' \itemize{
      #' \item \code{family} A string indicating a description of the error distribution. Options are 
      #' family = c("gaussian","binomial","Gamma","inverse.gaussian","poisson","quasi","quasibinomial","quasipoisson"). 
      #' The links can be also specified and can be found in \code{\link[stats]{family}}.
      #' \item \code{na.action} A function which indicates what should happen when the data contain NAs. 
      #' The default is set by the na.action setting of options, and is na.fail if that is unset. 
      #' The ‘factory-fresh’ default is na.omit. Another possible value is NULL, no action. Value na.exclude can be useful.
      #' }
    #' \item \code{fitting = "stepwise"} Indicates a stepwise regression via \code{\link[stats]{glm}} and \code{\link[stats]{step}}.
    #' The optional parameters are the same than for fitting = NULL. Stepwise can be performed backward or forward, as well as we can limit
    #' the number of steps. This can be done by the additional optional parameter \code{stepwise.arg}, which is a list contatining two parameters that belong 
    #' to \code{\link[stats]{step}}: steps and direction. An example would be: stepwise.arg = list(steps = 5, direction = "backward"). Default is NULL what indicates an unlimited forward stepwise search.
    #' \item \code{fitting = c("L1","L2","L1L2","gLASSO")}. These four options refer to ridge regression (L1 penalty), lasso regression (L2 penalty),
    #' elastic-net regression (L1L2 penalty) and group Lasso regression (group L2 penalty). The model is fitted via 
    #' \code{\link[glmnet]{glmnet}} and the corresponding penalties are found via \code{\link[glmnet]{cv.glmnet}}. This function \code{\link[glmnet]{glmnet}}
    #' forces by default to standardize predictors, however we have changed it to standardize = FALSE, and standardization should be done prior to 
    #' the downscaling process. 
    #' The optional parameters when fitting = c("L1","L2","L1L2","gLASSO") are:
    #' \itemize{
      #' \item \code{family} A string indicating a description of the error distribution. Options are 
      #' family = c("gaussian","binomial","Gamma","inverse.gaussian","poisson","quasi","quasibinomial","quasipoisson"). 
      #' The links CAN NOT be specified as the \code{\link[glmnet]{glmnet}} has not been programmed to handle links.
      #' However, the default ones can be found in \code{\link[stats]{family}}. If fitting = "gLASSO" then family must be "mgaussian".
      #' \item \code{offset} A vector of length nobs that is included in the linear predictor (a nobs x nc matrix for the "multinomial" family). 
      #' Useful for the "poisson" family (e.g. log of exposure time), or for refining a model by starting at a current fit. 
      #' Default is NULL. If supplied, then values must also be supplied to the predict function.
      #' }
   #' }
#' There are two things to consider. 
#' 1) If family = "binomial" then type = "response" when predicting values.
#' 2) Except for fitting = "MP", for the rest of the fitting options, the parameter site must be TRUE, unless 
#' we want a gLASSO, in this case site must be FALSE.
#' 
#' }
#' 
#' \strong{Neural Networks}
#' Neural network is based on the library \pkg{deepnet}. The optional parameters corresponds to those in \code{\link[deepnet]{nn.train}}
#' and are: \code{initW} = NULL, \code{initB} = NULL, \code{hidden} = c(10), \code{activationfun} = "sigm", \code{learningrate} = 0.001, \code{momentum} = 0.5, 
#' \code{learningrate_scale} = 1, \code{output} = "sigm", \code{numepochs} = 5000, \code{batchsize} = 100, \code{hidden_dropout} = 0, \code{visible_dropout} = 0. The values indicated are the default values.
#' 
#' \strong{Help}
#' 
#' If there are still doubts about the optional parameters despite the description here, we encourage to look for further details in the atomic functions: 
#' \code{\link[downscaleR]{analogs.train}}, \code{\link[downscaleR]{glm.train}} and \code{\link[deepnet]{nn.train}}.
#' 
#' @return A list of objects that contains the prediction on the train dataset and the model.
#' \itemize{
#'    \item \code{pred}: An object with the same structure as the predictands input parameter, but with pred$Data being the predictions and not the observations.
#'    \item \code{model}: A list with the information of the model: method, coefficients, fitting ...
#'    }
#' \href{https://github.com/SantanderMetGroup/downscaleR/wiki/training-downscaling-models}{downscaleR Wiki} for downscaling seasonal forecasting and climate projections.
#' @importFrom transformeR isRegular
#' @author J. Bano-Medina
#' @family downscaling.functions
#' @importFrom transformeR gridArithmetics
#' @importFrom sticky sticky
#' @export
#' @examples \donttest{
#' # Loading data
#' require(transformeR)
#' require(climate4R.datasets)
#' data("VALUE_Iberia_tas")
#' y <- VALUE_Iberia_tas
#' data("NCEP_Iberia_hus850", "NCEP_Iberia_psl", "NCEP_Iberia_ta850")
#' x <- makeMultiGrid(NCEP_Iberia_hus850, NCEP_Iberia_psl, NCEP_Iberia_ta850)
#' # Preparing the predictors
#' data <- prepareData(x = x, y = y, spatial.predictors = list(v.exp = 0.95))
#' # Training downscaling methods
#' model.analogs <- downscaleTrain(data, method = "analogs", n.analogs = 1)
#' model.regression <- downscaleTrain(data, method = "GLM",family = gaussian)
#' model.nnets <- downscaleTrain(data, method = "NN", hidden = c(10,5), output = "linear")
#' # Plotting the results for station 5
#' plot(y$Data[,5],model.analogs$pred$Data[,5], xlab = "obs", ylab = "pred")}

downscaleTrain <- function(obj, method, condition = NULL, threshold = NULL, model.verbose = TRUE, predict = TRUE, simulate = FALSE, ...) {
  method <- match.arg(method, choices = c("analogs", "GLM", "NN"))
  if ( method == "GLM") {
    if (attr(obj, "nature") == "spatial+local") {
      site <- "mix"}
    else {
      site <- "single"
    }
  }
  if ( method == "NN" || method == "analogs") {
    if (attr(obj,"nature") == "local") {
      site <- "single"
    }
    else if (attr(obj,"nature") == "spatial+local") {
      site <- "mix"
    }
    else {
      if (length(getDim(obj$y)) == 1) {
        obj$y <- redim(obj$y, loc = TRUE, member = FALSE)
        site <- "single"
      }
      else{
        site <- "multi"
      }
    }
  }
  dimNames <- getDim(obj$y)
  pred <- obj$y
  
  if (!is.null(threshold) & is.null(condition)) condition = "GE"
  if (!is.null(condition)) {
    if (is.null(threshold)) stop("Please specify the threshold value with parameter 'threshold'")
    ineq <- switch(condition,
                   "GT" = ">",
                   "GE" = ">=",
                   "LT" = "<",
                   "LE" = "<=")
    
  }
  
  # if (!is.null(threshold)) {
  #   if (condition == "GE") {
  #     epsilon <- 0.001
  #     obj$y <- gridArithmetics(obj$y,threshold-epsilon,operator = "-")
  #   } else if (condition == "GT") {
  #     obj$y <- gridArithmetics(obj$y,threshold,operator = "-")
  #   }
  # }
  
  if (isRegular(obj$y)) {
    mat.y <- array3Dto2Dmat(obj$y$Data)
    mat.p <- matrix(data = NA, nrow = nrow(mat.y), ncol = ncol(mat.y))
    regular <- TRUE
  }
  else {
    mat.y <- obj$y$Data
    mat.p <- matrix(data = NA, nrow = nrow(pred$Data), ncol = ncol(pred$Data))
    regular <- FALSE  
  }
  
  # Multi-site
  if (site == "multi") {
    xx <- sticky(obj$x.global)
    yy <- mat.y
    dates.y <- getRefDates(obj$y)
    if (method == "analogs") {
      atomic_model <- downs.train(xx, yy, method, model.verbose, dates = dates.y, ...)
      atomic_model$dates$test <- getRefDates(obj$y)}
    else {      
      if (anyNA(yy)) {
        ind <- sapply(1:ncol(yy),FUN = function(z){which(is.na(yy[,z]))}) %>% unlist() %>% unique()
        message(paste(round(length(ind)/nrow(yy)*100,digits = 2),"% of observations contains NaN, removed from the training phase ..."))
        yy <- yy[-ind,,drop = FALSE]
        xx <- xx[-ind,,drop = FALSE]
        dates.y <- getRefDates(obj$y)[-ind]
      }
      atomic_model <- downs.train(xx, yy, method, model.verbose, ...)
    }
    if (isTRUE(predict)) mat.p <- as.matrix(downs.predict(obj$x.global, method, atomic_model, simulate))
  }
  # Single-site
  else if (site == "single") {
    stations <- dim(mat.p)[2]
    atomic_model <- vector("list",stations)
    for (i in 1:stations) {
      if (attr(obj,"nature") == "local") {
        xx = obj$x.local[[i]]$member_1
        attr(xx,"predictorNames") <- attr(obj$x.local,"predictorNames")
        xx %<>% sticky()                                  
      } else {
        xx = sticky(obj$x.global)
      }
      yy = mat.y[,i, drop = FALSE]
      if (all(is.na(yy))) {
        mat.p[,i] <- yy
      } else{
        if (is.null(condition)) {ind = eval(parse(text = "which(!is.na(yy))"))}
        else {ind = eval(parse(text = paste("yy", ineq, "threshold")))}
        if (anyNA(ind)) ind[which(is.na(ind))] <- FALSE
        if (method == "analogs") {
          atomic_model[[i]] <- downs.train(xx[ind,, drop = FALSE], yy[ind,,drop = FALSE], method, model.verbose, dates = getRefDates(obj$y)[ind], ...)
        } else {
          tryCatch({
            atomic_model[[i]] <- downs.train(xx[ind,, drop = FALSE], yy[ind,,drop = FALSE], method, model.verbose, ...)
          } , error = function(x) {
            warning(sprintf("... couldn't create model for point %d out of %d ...",
                            i, stations))
            # atomic_model[[i]] = NULL
          })
        }
        if (method == "analogs") {atomic_model[[i]]$dates$test <- getRefDates(obj$y)}
        if (isTRUE(predict)) {
          if (is.null(atomic_model[[i]])) {
            mat.p[,i] <- rep(NA, 1, nrow(mat.p))
          } else {
            mat.p[,i] <- downs.predict(xx, method, atomic_model[[i]], simulate)
          }
        }
      }
    }
  }
  # Mix - Global predictors with local predictors
  else if (site == "mix") {
    stations <- dim(mat.p)[2]
    atomic_model <- vector("list",stations)
    for (i in 1:stations) {
      xx1 = obj$x.local[[i]]$member_1
      xx2 = obj$x.global
      xx <- cbind(xx1,xx2)
      attr(xx,"predictorNames") <- c(attr(obj$x.local,"predictorNames"),attr(xx2,"predictorNames"))
      xx %<>% sticky()
      yy = mat.y[,i, drop = FALSE]
      if (all(is.na(yy))) {
        mat.p[,i] <- yy
      }
      else{
        if (is.null(condition)) {ind = eval(parse(text = "which(!is.na(yy))"))}
        else {ind = eval(parse(text = paste("yy", ineq, "threshold")))}
        if (anyNA(ind)) ind[which(is.na(ind))] <- FALSE
        if (method == "analogs") {
          atomic_model[[i]] <- downs.train(xx[ind,, drop = FALSE], yy[ind,,drop = FALSE], method, model.verbose, dates = getRefDates(obj$y)[ind], ...)}
        else {
          tryCatch ({
            atomic_model[[i]] <- downs.train(xx[ind,, drop = FALSE], yy[ind,,drop = FALSE], method, model.verbose, ...)
          } , error = function(x) {
            warning(sprintf("... couldn't create model for point %d out of %d ...",
                            i, stations))
            # atomic_model[[i]] = NULL
          })
        }
        if (method == "analogs") {atomic_model[[i]]$dates$test <- getRefDates(obj$y)}
        if (isTRUE(predict)) mat.p[,i] <- downs.predict(xx, method, atomic_model[[i]], simulate)}
    }
  }
  if (isTRUE(predict)) {
    if (regular) {
      pred$Data <- mat2Dto3Darray(mat.p, x = pred$xyCoords$x, y = pred$xyCoords$y)
    } 
    else {
      pred$Data <- mat.p
    }
    attr(pred$Data, "dimensions") <- dimNames
  } else {
    pred <- NULL
  }
  model <- list("pred" = pred, "model" = list("method" = method, "site" = site, "atomic_model" = atomic_model, "condition" = condition, "threshold" = threshold))
  return(model)
}

##############################################################################################################
#                     DOWNSCALING                                                                            #
##############################################################################################################
#' @title Switch to selected downscale method.
#' @description Internal function of \code{\link[downscaleR]{downscaleTrain}} that switches to the selected method.
#' @param x The input grid. Class: matrix.
#' @param y The observations dataset. Class: matrix.
#' @param method Type of transer function. Options are: analogs, GLM and NN. 
#' @param model.verbose Logic value. Indicates wether the information concerning the model infered is limited to the 
#' essential information (model.verbose = FALSE)  or a more detailed information (model.verbose = TRUE, DEFAULT). This is
#' recommended when you want to save memory. Only operates for GLM.
#' @param ... Optional parameters. These parameters are different depending on the method selected. Every parameter has a default value set in the atomic functions in case that no selection is wanted. For this reason see the atomic functions for more details: \code{\link[downscaleR]{glm.train}} and \code{\link[deepnet]{nn.train}}.  
#' @return An object with the information of the selected model.
#' @details The optional parameters of neural networks can be found in the library \pkg{deepnet} via \code{\link[deepnet]{nn.train}}This function is internal and should not be used by the user. The user should use \code{\link[downscaleR]{downscaleTrain}}.
#' @author J. Bano-Medina
#' @importFrom deepnet nn.train

downs.train <- function(x, y, method, model.verbose = TRUE, ...) {
  if (method == "NN") {
    arglist <- list(...)
    arglist[["x"]] <- x
    arglist[["y"]] <- y
    if (is.null(arglist[["numepochs"]])) arglist[["numepochs"]] <- 5000
    if (is.null(arglist[["learningrate"]])) arglist[["learningrate"]] <- 0.001
  }
  
  switch(method,
         "analogs" = atomic_model <- analogs.train(x, y, ...),
         "GLM"     = atomic_model <- glm.train(x, y, model.verbose = model.verbose,...),
         "NN"      = atomic_model <- do.call("nn.train",arglist))
  
  
  
  return(atomic_model)}
SantanderMetGroup/downscaleR documentation built on July 4, 2023, 4:28 a.m.