diff --git a/.travis.yml b/.travis.yml index aaa6fb3..2d690e0 100644 --- a/.travis.yml +++ b/.travis.yml @@ -4,6 +4,9 @@ language: R sudo: true cache: packages +#remove this line when this pull request is closed +# https://github.com/travis-ci/travis-build/pull/1553 +fortran: false bioc_required: true bioc_use_devel: true @@ -19,8 +22,8 @@ r: r_github_packages: - - hadley/devtools#1263 - - rte-antares-rpackage/antaresRead@next_release + - rte-antares-rpackage/antaresRead@master +# - hadley/devtools#1263 #before - Bioconductor-mirror/zlibbioc # - Bioconductor-mirror/rhdf5 #after diff --git a/DESCRIPTION b/DESCRIPTION index 5c2a5ac..be88ae4 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -1,7 +1,7 @@ Package: antaresProcessing Type: Package Title: 'Antares' Results Processing -Version: 0.16.0 +Version: 0.16.0.902 Date: 2018-09-28 Authors@R: c( person("Jalal-Edine", "ZAWAM", , "jalal-edine.zawam@rte-france.com", role = c("aut", "cre")), @@ -23,7 +23,7 @@ BugReports: https://github.com/rte-antares-rpackage/antaresProcessing/issues License: GPL (>= 2) | file LICENSE LazyData: TRUE Depends: antaresRead (>= 1.1.5) -Imports: data.table, methods, stats +Imports: data.table, methods, stats, stringi Suggests: rhdf5 (>= 2.24.0), parallel, testthat, knitr, rmarkdown, covr RoxygenNote: 6.1.0 VignetteBuilder: knitr diff --git a/NAMESPACE b/NAMESPACE index 224ca90..c454f99 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -1,18 +1,25 @@ # Generated by roxygen2: do not edit by hand export(.setAliasH5) +export(addConvergencePriceArea) +export(addConvergencePriceSystem) export(addDownwardMargin) export(addExportAndImport) export(addLoadFactorLink) +export(addMonotones) +export(addNeighbours) export(addNetLoad) export(addProcessingH5) export(addUpwardMargin) export(compare) +export(correctBalance) export(externalDependency) +export(getAllNeighbours) export(getValues) export(loadFactor) export(mergeAllAntaresData) export(modulation) +export(neighbours) export(netLoadRamp) export(surplus) export(surplusClusters) @@ -26,3 +33,4 @@ importFrom(stats,as.formula) importFrom(stats,median) importFrom(stats,quantile) importFrom(stats,sd) +importFrom(stringi,stri_detect_fixed) diff --git a/NEWS b/NEWS index eaf008d..4f175d2 100644 --- a/NEWS +++ b/NEWS @@ -1,5 +1,14 @@ Copyright © 2016 RTE Réseau de transport d’électricité +Changes in version 0.17.0 (2018-11-08) + +NEW FEATURES: +* New function addMonotones() computes monotones and add it to the input (#37). +* New function correctBalance() corrects the BALANCE with 'ROW BAL' (#33). +* New functions neighbours(), addNeighbours() and getAllNeighbours() to get neighbours (#21). +* New functions addConvergencePriceSystem() computes the biggest system without congestion (#21). +* New functions addConvergencePriceArea() computes the biggest system without congestion for each area (#21). + Changes in version 0.16.0 (2018-09-28) NEW FEATURES: diff --git a/R/addConvergencePriceArea.R b/R/addConvergencePriceArea.R new file mode 100644 index 0000000..8e98251 --- /dev/null +++ b/R/addConvergencePriceArea.R @@ -0,0 +1,109 @@ +#' addConvergencePriceArea +#' +#' This function computes priceConvergenceArea, priceConvergenceArea represent +#' the biggest system without congestion for one area. +#' +#' @param antaresData Object of class \code{antaresData} created with function +#' \code{\link[antaresRead]{readAntares}}. antaresData must contains areas and +#' links details hourly data with linkCapacity. +#' @examples +#' \dontrun{ +#' +#' myData <- readAntares(areas = "all", +#' links = "all", +#' showProgress = FALSE, +#' linkCapacity = TRUE, +#' mcYears = "all") +#' +#' myDataRV <- removeVirtualAreas(x = myData, +#' storageFlexibility = getAreas(c("psp", "hub")), +#' production = getAreas("off")) +#' +#' addConvergencePriceArea(myData) +#' } +#' @importFrom stringi stri_detect_fixed +#' @export +addConvergencePriceArea <- function(antaresData = NULL){ + .check_x(antaresData) + if((!is(antaresData, "antaresDataList")) | !("areas" %in% names(antaresData))){ + stop("Import areas data. antaresData must be an antaresDataList.") + } + if(attr(antaresData, "synthesis") | attr(antaresData, "timeStep") != "hourly"){ + stop("Import hourly details data") + } + + if(is.null(antaresData$areas$neighbours)){ + addNeighbours(antaresData) + } + if(is.null(antaresData$areas$priceConvergenceSystem)){ + addConvergencePriceSystem(antaresData) + } + #pb with R check, init variable + priceConvergenceAreaN <- NULL + priceConvergenceAreaN1 <- NULL + priceConvergenceSystem <- NULL + priceConvergenceArea <- NULL + neighboursN <- NULL + neighboursN1 <- NULL + + #priceConvergenceArea = area + antaresData$areas[, priceConvergenceAreaN := as.character(area), + by=.(area, mcYear, timeId)] + antaresData$areas[, ':=' (neighboursN = neighbours)] + + #we must get the virtual areas + resAttr <- attributes(antaresData) + virtualAreas <- resAttr$virtualNodes$storageFlexibility + virtualAreas <- c(virtualAreas, resAttr$virtualNodes$production) + + #priceConvergenceArea = area + neighbours in priceConvergenceSystem if area in priceConvergenceSystem + #init + antaresData$areas[, priceConvergenceAreaN1 := priceConvergenceAreaN] + indexToEdit <- antaresData$areas[stringi::stri_detect_fixed(pattern = area, + str = priceConvergenceSystem), + which=TRUE] + + antaresData$areas[indexToEdit, ':=' (priceConvergenceAreaN1 = paste(sort(unique(c(intersect(strsplit(priceConvergenceSystem, split = " ")[[1]], + strsplit(neighboursN, split = " ")[[1]]), + strsplit(priceConvergenceAreaN, split = " ")[[1]]))), + collapse = " ")), + by = .(priceConvergenceSystem, priceConvergenceAreaN)] + + #for all index where priceConvergenceArea != priceConvergenceSystem + #then test if neighbours of neighbours are in priceConvergenceSystem + #init system + antaresData$areas[, ':=' (neighboursN = neighbours)] + antaresData$areas[, ':=' (neighboursN1 = neighbours)] + + #antaresData$areas[, priceConvergenceAreaN1 := priceConvergenceAreaN] + indexWhereSomethingChange <- c(1, 2) + i <- 0 + while (length(indexWhereSomethingChange) > 0) { + indexWhereSomethingChange <- antaresData$areas[priceConvergenceAreaN!=priceConvergenceAreaN1, which=TRUE] + + # iterate N1 := N + antaresData$areas[indexWhereSomethingChange, priceConvergenceAreaN := priceConvergenceAreaN1] + antaresData$areas[indexWhereSomethingChange, ':=' (neighboursN = neighboursN1)] + + #compute N1 + antaresData$areas[indexWhereSomethingChange, ':=' (neighboursN1 = paste(getAllNeighbours(neighboursN, + virtualAreas = virtualAreas), + collapse = " ")), + by = .(neighboursN)] + antaresData$areas[indexWhereSomethingChange, ':=' (priceConvergenceAreaN1 = paste(sort(unique(c(intersect(strsplit(priceConvergenceSystem, split = " ")[[1]], + strsplit(neighboursN1, split = " ")[[1]]), + strsplit(priceConvergenceAreaN, split = " ")[[1]]))), + collapse = " ")), + by = .(priceConvergenceSystem, priceConvergenceAreaN)] + + i <- i +1 + } + + antaresData$areas[, priceConvergenceArea := priceConvergenceAreaN1] + + antaresData$areas[, c("priceConvergenceAreaN", + "priceConvergenceAreaN1", + "neighboursN", + "neighboursN1") := NULL] + invisible(antaresData) +} diff --git a/R/addConvergencePriceSystem.R b/R/addConvergencePriceSystem.R new file mode 100644 index 0000000..d5f54cd --- /dev/null +++ b/R/addConvergencePriceSystem.R @@ -0,0 +1,62 @@ +#' addConvergencePriceSystem +#' +#' This function computes priceConvergenceSystem, priceConvergenceSystem represent +#' the biggest system without congestion. +#' +#' @param antaresData Object of class \code{antaresData} created with function +#' \code{\link[antaresRead]{readAntares}}. antaresData must contains areas and +#' links details data with linkCapacity. +#' @examples +#' \dontrun{ +#' +#' myData <- readAntares(areas = "all", +#' links = "all", +#' showProgress = FALSE, +#' linkCapacity = TRUE, +#' mcYears = "all") +#' +#' myDataRV <- removeVirtualAreas(x = myData, +#' storageFlexibility = getAreas(c("psp", "hub")), +#' production = getAreas("off")) +#' +#' addConvergencePriceSystem(myData) +#' } +#' @export +addConvergencePriceSystem <- function(antaresData = NULL){ + .check_x(antaresData) + if((!is(antaresData, "antaresDataList")) | !("areas" %in% names(antaresData)) | + !("links" %in% names(antaresData)) ){ + stop("Import areas and links data.") + } + if(attr(antaresData, "synthesis") | attr(antaresData, "timeStep") != "hourly"){ + stop("Import hourly details data") + } + + if(is.null(antaresData$links$congestion)){ + antaresData <- addLoadFactorLink(antaresData) + } + #R CMD CHECK, init some variables + congestion <- NULL + priceConvergenceArea <- NULL + areasConver <- NULL + allAreasTgList <- NULL + i.areasConver <- NULL + + + antaresData$links[, ':=' (from = strsplit(link, split = " - " )[[1]][1], + to = strsplit(link, split = " - " )[[1]][2]), + by = .(link)] + dontSeperate <- antaresData$links[congestion==0, .(mcYear, timeId, from, to)] + + converAreas <- dontSeperate[, list(allAreasTgList = list(c(from,to))) , + by=.(mcYear, timeId)] + + converAreas[ , areasConver := as.vector(paste(sort(unique(allAreasTgList[[1]])), + collapse = " ")), + by=.(mcYear, timeId)] + + antaresData$areas[converAreas,(paste("priceConvergenceSystem")) := i.areasConver, + on=.(mcYear, timeId)] + + invisible(antaresData) +} diff --git a/R/addMonotones.R b/R/addMonotones.R new file mode 100644 index 0000000..24e447f --- /dev/null +++ b/R/addMonotones.R @@ -0,0 +1,123 @@ +#Copyright © 2016 RTE Réseau de transport d’électricité + +#' addMonotones +#' +#' This function computes monotones for some variables. +#' @param antaresData Object of class \code{antaresData} created with function +#' \code{\link[antaresRead]{readAntares}}. +#' @param variable An ANTARES variable. +#' +#' @return +#' \code{addMonotones} modifies its input by adding monotones. +#' +#' +#' @examples +#' \dontrun{ +#' # First simulation +#' studyPath <- "path/to/study/" +#' +#' setSimulationPath(studyPath, 1) +#' myData1 <- readAntares(areas = "all", +#' districts = "all", synthesis = FALSE) +#' addMonotones(antaresData = myData1, +#' variable = "LOAD") +#' +#' } +#' +#' @export +#' +addMonotones <- function(antaresData = NULL, variable = NULL){ + + .check_x(antaresData) + + if(attr(antaresData, "synthesis")){ + stop("antaresData are synthesis, addMonotones() needs detail data.") + } + + if(!is.character(variable)){ + stop("variable is not a character") + } + + if(is(antaresData, "antaresDataList")){ + #chack if variable is correct + nameCol <- NULL + for(i in 1:length(antaresData)){ + nameCol <- c(nameCol, names(antaresData[[i]])) + } + if(!(variable %in% nameCol)){ + stop("incorrect variable") + } + #call antaresData for an antaresDataTable + for(na in names(antaresData)){ + if(variable %in% names(antaresData[[na]])){ + addMonotones(antaresData[[na]], variable = variable) + } + } + }else{ + if(!(variable %in% names(antaresData))){ + stop("Incorrect variable") + } + classVariable <- class(antaresData[[variable]]) + if(classVariable %in% c("factor", "POSIXt", "character")){ + stop("Incorrect variable class") + }else if(classVariable=="numeric"){ + funcClass <- as.numeric + digitMonoMean <- 2 + }else if(classVariable=="integer"){ + funcClass <- as.integer + digitMonoMean <- 0 + }else if(classVariable=="double"){ + funcClass <- as.double + digitMonoMean <- 2 + }else{ + stop("wrong class variable") + } + + if(length(unique(antaresData$mcYear)) <= 1){ + stop("antaresData must contain at least two mcYears.") + } + #variable to take from x + idCols <- .idCols(antaresData) + varToTake <- variable + + antaresDataCopy <- copy(antaresData[, mget(c(idCols, varToTake))]) + #by of x + getByA <- setdiff(getIdCols(antaresDataCopy), pkgEnv$idTimeVars) + + #get only what you want + nameNewColumn <- paste0(varToTake, "Mono") + antaresDataCopy[, (nameNewColumn) := funcClass(0)] + #maybe faster but we must deal with column/string + # resMono <- antaresData[, eval(quote(.(nameNewColumn = + # sort(get(varToTake), + # decreasing = TRUE)))), + # by = getByA] + antaresDataCopy <- antaresDataCopy[, (nameNewColumn) := sort(get(varToTake), + decreasing = TRUE), + by = getByA] + antaresDataCopy[, c(varToTake) := NULL] + #Cast the result to be able to make a rowMean + IdColsWMcYear <- idCols[idCols!="mcYear"] + myFormula <- sprintf("%s ~ mcYear", paste(IdColsWMcYear, collapse = "+")) + antaresDataCopy <- data.table::dcast(data = antaresDataCopy, + formula = as.formula(myFormula), + value.var = c(paste0(varToTake, "Mono"))) + newNameColumnnMonoMc <- paste0(varToTake,"MonoMc" , unique(antaresData$mcYear)) + data.table::setnames(x = antaresDataCopy, + old = as.character(unique(antaresData$mcYear)), + new = newNameColumnnMonoMc) + #make a monotone rowMean + nameNewColum <- c(paste0(varToTake, "MonoMean")) + antaresDataCopy[, (nameNewColum) := funcClass(0)] + antaresDataCopy <- antaresDataCopy[, (nameNewColum) := funcClass(round(rowMeans(.SD), digitMonoMean)), + by = IdColsWMcYear, .SDcols = newNameColumnnMonoMc] + #antaresData <- merge(antaresData, resMonoMean, by = IdColsWMcYear) + colToAdd <- setdiff(names(antaresDataCopy), IdColsWMcYear) + antaresData[antaresDataCopy, + (colToAdd) := mget(paste0("i.", colToAdd)), + on = IdColsWMcYear] + invisible(antaresData) + } +} + + diff --git a/R/addUpwardMargin.R b/R/addUpwardMargin.R index e1c3d46..8575cd8 100644 --- a/R/addUpwardMargin.R +++ b/R/addUpwardMargin.R @@ -5,8 +5,7 @@ #' This function computes isolated and interconnected upward margins of areas and #' add them to an antaresData object. #' -#' @param x An object of class \code{antaresData} created with -#' \code{\link[antaresRead]{readAntares}} +#' @param x An object of class \code{\link[antaresRead]{readAntares}} (or \code{\link[antaresRead]{simOptions}}) created with 'readAntares()' (or 'setSimulationPath()') #' #' @return #' The function modifies its input by adding to it two new columns diff --git a/R/correctBalance.R b/R/correctBalance.R new file mode 100644 index 0000000..c2861b1 --- /dev/null +++ b/R/correctBalance.R @@ -0,0 +1,154 @@ +#Copyright © 2016 RTE Réseau de transport d’électricité + +#' correctBalance +#' +#' This function corrects the BALANCE with 'ROW BAL.'. +#' +#' @return +#' \code{correctBalance} modifies its input by editing BALANCE and 'ROW BAL.'. +#' Formulas : +#' \enumerate{ +#' \item \eqn{BALANCE = BALANCE - 'ROW. BAL.'} +#' \item \eqn{ROW. BAL = 0} +#' } +#' +#' @inheritParams addUpwardMargin +#' +#' @examples +#' \dontrun{ +#' # First simulation +#' studyPath <- "path/to/study/" +#' +#' setSimulationPath(studyPath, 1) +#' mydata1 <- readAntares(areas = "all", districts = "all", synthesis = FALSE) +#' correctBalance(mydata1) +#' +#' } +#' +#' @export +#' +correctBalance <- function(x) { + .check_x(x) + + if(.isSimOpts(x)){ + .correctBalanceOpts(x) + return(invisible()) + }else{ + if (is(x, "antaresDataList")) { + if(!is.null(x$areas)){ + x$areas <- .correctBalanceAreas(x$areas) + } + if(!is.null(x$districts)){ + x$districts <- .correctBalanceAreas(x$districts) + } + }else{ + if("area" %in% names(x) | "district" %in% names(x)){ + x <- .correctBalanceAreas(x) + } + } + } + + return(x) + +} + +.correctBalanceAreas <- function(x){ + .check_x(x) + oldBalance <- NULL + x[, oldBalance := BALANCE] + x[, BALANCE := BALANCE - `ROW BAL.`] + x[, `ROW BAL.` := 0] + + return(x) + +} + +.correctBalanceOpts <- function(optsP = NULL){ + + sapply(pkgEnv$timeStepPosible, function(timeStepS){ + for(mcYearIns in c("mcInd", "mcAll")){ + suppressWarnings(addProcessingH5(opts = optsP, + mcY = mcYearIns, + timeStep = timeStepS, + evalAreas = list( + oldBalance = "BALANCE" + ), + evalDistricts = list( + oldBalance = "BALANCE" + ))) + #read the data + if(mcYearIns=="mcInd"){ + mcYearRead <- "all" + }else{ + mcYearRead <- NULL + } + dataRead <- readAntares(opts = optsP, + areas = "all", + districts = "all", + select = c("BALANCE", "ROW BAL."), + mcYears = mcYearRead, + timeStep = timeStepS) + #correct the data + dataCor <- correctBalance(x = dataRead) + #write the corrected data + .writeH5Antares(dataC = dataCor, opts = optsP, timeStepC = timeStepS) + } + }) +} + +.writeH5Antares <- function(dataC = NULL, opts = NULL, timeStepC = NULL){ + #loop data + for(typeData in c("areas", "districts")){ + if(typeData=="district"){ + print("test") + } + dataToC <- dataC[[typeData]] + ifelse(typeData=="areas", keyData <- "area", keyData <- "district") + if(!is.null(dataToC)){ + keyToW <- unique(dataToC[[keyData]]) + #correct only the values + varToCorrect <- setdiff(names(dataToC), .idCols(dataToC)) + if(attr(dataC, "synthesis")){ + sapply(varToCorrect, function(varS){ + sapply(keyToW, function(keyToS){ + timeIdW <- unique(dataToC[get(keyData)==keyToS, timeId]) + dataCVar <- dataToC[get(keyData)==keyToS, get(varS)] + .h5Antares_edit_variable( + pathH5 = opts$h5path, + instanceData = keyToS, + classData = typeData, + timeId = timeIdW, + antVar = varS, + newValue = dataCVar, + mcYear = NULL, + allTimeId = TRUE, + timeStep = timeStepC + ) + }) + }) + }else{ + mcYearW <- unique(dataToC$mcYear) + sapply(X = mcYearW, FUN = function(mcYearS){ + sapply(varToCorrect, function(varS){ + sapply(keyToW, function(keyToS){ + timeIdW <- unique(dataToC[get(keyData)==keyToS, timeId]) + dataCVar <- dataToC[get(keyData)==keyToS & mcYear == mcYearS, + get(varS)] + .h5Antares_edit_variable( + pathH5 = opts$h5path, + instanceData = keyToS, + classData = typeData, + timeId = timeIdW, + antVar = varS, + newValue = dataCVar, + mcYear = mcYearS, + allTimeId = TRUE, + timeStep = timeStepC + ) + }) + }) + }) + } + } + } +} diff --git a/R/h5_processing.R b/R/h5_processing.R index 6498d1a..7078714 100644 --- a/R/h5_processing.R +++ b/R/h5_processing.R @@ -10,6 +10,7 @@ #' @param addUpwardMargin \code{boolean} refer to \link[antaresProcessing]{addUpwardMargin} #' @param addExportAndImport \code{boolean} refer to \link[antaresProcessing]{addExportAndImport} #' @param addLoadFactorLink \code{boolean} refer to \link[antaresProcessing]{addLoadFactorLink} +#' @param correctBalance \code{boolean} refer to \link[antaresProcessing]{correctBalance} #' @param externalDependency \code{boolean} refer to \link[antaresProcessing]{externalDependency} #' @param loadFactor \code{boolean} refer to \link[antaresProcessing]{loadFactor} #' @param modulation \code{boolean} refer to \link[antaresProcessing]{modulation} @@ -54,6 +55,7 @@ #' addUpwardMargin = TRUE, #' addExportAndImport = TRUE, #' addLoadFactorLink = TRUE, +#' correctBalance = TRUE, #' externalDependency = TRUE, #' loadFactor = TRUE, #' modulation = TRUE, @@ -83,6 +85,7 @@ addProcessingH5 <- function(opts = simOptions(), addUpwardMargin = FALSE, addExportAndImport = FALSE, addLoadFactorLink = FALSE, + correctBalance = FALSE, externalDependency = FALSE, loadFactor = FALSE, modulation = FALSE, @@ -104,6 +107,12 @@ addProcessingH5 <- function(opts = simOptions(), stop("opts not refear to an h5 file") } + if(correctBalance){ + try({ + correctBalance(x = opts) + }) + } + allData <- allProcess if(allData){ addNetLoad <- TRUE @@ -395,7 +404,7 @@ addProcessingH5 <- function(opts = simOptions(), links = ln, clusters = clu, districts = dr, - opts = opts, select = c(select,columnsToSelects), + opts = opts, select = c(select, columnsToSelects), mcYears = mcYears, timeStep = timeStep) res <- as.antaresDataList(res) @@ -755,6 +764,7 @@ addProcessingH5 <- function(opts = simOptions(), res$clusters <- merge(res$clusters, surplusClusters, by = idC) }) } + options(warn = oldw) res } diff --git a/R/neighbours.R b/R/neighbours.R new file mode 100644 index 0000000..cf37e4a --- /dev/null +++ b/R/neighbours.R @@ -0,0 +1,83 @@ +#' neighbours +#' +#' This function return a list of neighbours. +#' @param antaresData Object of class \code{antaresData} created with function +#' \code{\link[antaresRead]{readAntares}}. +#' @param areas A vector with several areas names. +#' @param virtualAreas A vector with several virtual areas names. +#' @param areasString A string with several areas names separated by an espace, +#' see the examples. +#' +#' @return +#' \code{neighbours} return a vector with neighbours areas names. +#' \code{addNeighbours} modifies its input by adding a column neighbours. +#' \code{getAllNeighbours} return a vector with neighbours areas names. +#' +#' @examples +#' \dontrun{ +#' +#' res <- neighbours(areas = c("a", "c"), +#' virtualAreas = getAreas("psp")) +#' +#' myData <- readAntares(areas = c("a", "c"), links = getLinks("a"), +#' showProgress = FALSE) +#' +#' addNeighbours(myData) +#' +#' res <- getAllNeighbours(areasString = "a b") +#' } +#' @export +neighbours <- function(areas = NULL, virtualAreas = NULL){ + res <- NULL + res <- sapply(as.character(areas), FUN = function(myArea){ + linksNeighbours <- getLinks(areas = myArea, exclude = virtualAreas) + setdiff(unique(unlist(strsplit(linksNeighbours, split = " - " ))), myArea) + }) + #init + neighbourDT <- c(1) + if(is.null(names(res))){ + DF <- as.data.frame(res) + areasNamesDT <- names(DF) + neighbourDT <- data.table::data.table(area = areasNamesDT) + neighbourDT[, neighbours := paste(DF[[area]], collapse=" "), by=.(area)] + }else{ + areasNamesDT <- names(res) + neighbourDT <- data.table::data.table(area = areasNamesDT) + neighbourDT[, neighbours := paste(res[[area]], collapse=" "), by=.(area)] + } + + return(neighbourDT) +} + +#' @rdname neighbours +#' @export +addNeighbours <- function(antaresData = NULL){ + .check_x(antaresData) + if(!(("area" %in% names(antaresData)) | ("areas" %in% names(antaresData))) ){ + stop("Import areas data.") + } + + if(!is.null(antaresData$areas)){ + addNeighbours(antaresData$areas) + return(invisible(antaresData)) + } + #R CMD Check, null variable + i.neighbours <- NULL + + resAttr <- attributes(antaresData) + virtualAreas <- resAttr$virtualNodes$storageFlexibility + + resNeigh <- neighbours(areas = unique(antaresData$area), virtualAreas = virtualAreas) + antaresData[resNeigh, neighbours := i.neighbours, + on=.(area)] + + invisible(antaresData) +} + +#' @rdname neighbours +#' @export +getAllNeighbours <- function(areasString = NULL, virtualAreas = NULL){ + res <- unlist(strsplit(areasString, split = " ")); + neighS <- neighbours(res, virtualAreas = virtualAreas); + unique(unlist(strsplit(neighS[[2]], split = " "))) +} diff --git a/R/zzz.R b/R/zzz.R index c312586..666b979 100644 --- a/R/zzz.R +++ b/R/zzz.R @@ -80,6 +80,12 @@ c("areas", "clusters", "mcYears", "WIND", "SOLAR", "H. ROR", "H. STOR", "MRG. PRICE") ) + + setAlias( + "correctBalance", + "Data required by function 'correctBalance()'", + c("areas", "BALANCE", "ROW BAL.") + ) sapply(names(pkgEnv$process), function(X){ tpAlias <- pkgEnv$process[[X]] X <- paste0("Out_", X) @@ -116,6 +122,14 @@ globalVariables( .idCols <- antaresRead:::.idCols .addClassAndAttributes <- antaresRead:::.addClassAndAttributes .groupByDistrict <- antaresRead:::.groupByDistrict +#TODO add after antaresRead 2.2.2 +#.check_x <- antaresRead:::.check_x +#.h5Antares_edit_variable <- antaresRead:::.h5Antares_edit_variable +#.skipFunctionH5 <- antaresRead:::.skipFunctionH5 +#.check_if_h5_is_in_tmp <- antaresRead:::.check_if_h5_is_in_tmp +#.isSimOpts <- antaresRead:::.isSimOpts +.get_by_area <- antaresRead:::.get_by_area +.get_by_link <- antaresRead:::.get_by_link pkgEnv <- antaresRead:::pkgEnv @@ -127,6 +141,8 @@ pkgEnv <- antaresRead:::pkgEnv pkgEnv$process$addNetLoad$areas <- c("netLoad") pkgEnv$process$addNetLoad$districts <- c("netLoad") +pkgEnv$process$correctBalance$areas <- c("oldBalance") +pkgEnv$process$correctBalance$districts <- c("oldBalance") pkgEnv$process$addDownwardMargin$areas <- c("isolatedDownwardMargin", "interconnectedDownwardMargin") @@ -217,3 +233,204 @@ pkgEnv$processDispo <- data.frame( "surplusClusters" )) + + +.check_if_h5_is_in_tmp<-function(h5filePath=NULL, path=NULL, stop=FALSE, printMessage=TRUE){ + + resH5NotInTmp<-!grepl("Temp", h5filePath, ignore.case = TRUE) & !grepl("tmp", h5filePath, ignore.case = TRUE) + if(resH5NotInTmp){ + if(printMessage){ + print(paste0("h5file : ", h5filePath)) + print(paste0("path : ", path)) + } + }else{ + return(TRUE) + } + + messageToPrint<-"h5file is not in temp folder" + if(stop & resH5NotInTmp){ + stop(messageToPrint) + } + if(resH5NotInTmp){ + if(printMessage){ + message(messageToPrint) + } + } + + return(FALSE) +} + +.skipFunctionH5<-function(h5file = NULL){ + testthat::skip_if(is.null(h5file), "h5file is null") + if(!is.null(h5file)){ + testthat::skip_if(!.check_if_h5_is_in_tmp(h5file), "h5file is not in temp folder") + } + testthat::skip_if_not(.requireRhdf5_Antares(stopP = FALSE), + "please install a correct rhdf5 version") + + #I dont why but according to CRAN Team antaresProcessing try to write in the user library + # but there are checks to verify that h5 file is in tmp folder + #to comment in the futur + testthat::skip_on_cran() + #TODO DEL + testthat::skip("h5") +} + +.cleanDataCluster <- function(clDes = NULL){ + for(columnC in names(clDes)){ + if(class(clDes[[columnC]]) %in% c("double", "numeric", "integer")){ + clDes[is.na(get(columnC)), (columnC) := 0] + } + } + nameMustRun <- "must.run" + if(nameMustRun %in% names(clDes)){ + clDes[is.na(get("must.run")), ("must.run") := FALSE] + }else{ + clDes[, (nameMustRun) := FALSE] + } + return(clDes) +} + +#TODO DEL after antaresRead 2.2.2 +#' Test antaresData +#' +#' @param x if x is antaresData class +#' +#' @noRd +.isAntaresData <- function(x){ + "antaresData" %in% class(x) +} + +#TODO DEL after antaresRead 2.2.2 +.check_x <- function(x = NULL){ + if ("list" %in% class(x)){ + for (elementI in x){ + .check_x(elementI) + } + }else{ + if (!(.isSimOpts(x) | .isAntaresData(x))){ + stop(paste0(substitute(x), + " should be an object of class 'antaresData' (or 'simOptions') created with 'readAntares()' (or 'setSimulationPath()')")) + }else{ + return(TRUE) + } + } +} + +#TODO DEL after antaresRead 2.2.2 +#' edit h5 file for TEST +#' currently for all data except clusters +#' +#' @param pathH5 path H5 file +#' @param instanceData character name of one instance +#' @param classData character name of class instance +#' @param timeId timeId to change +#' @param antVar antares Variable to change +#' @param newValue the newValue +#' +#' @noRd +.h5Antares_edit_variable <- function(pathH5 = NULL, + instanceData = NULL, + classData = NULL, + timeId = 1, + antVar = NULL, + newValue = NULL, + mcYear = NULL, + allTimeId = FALSE, + timeStep = "hourly"){ + + if (is.null(instanceData) | is.null(classData)){ + stop("instanceData and classData must be set together") + } + + if (classData=="areas"){ + classDataS <- "area" + }else if(classData=="links"){ + classDataS <- "link" + }else{ + classDataS <- "district" + } + + if (is.null(mcYear)){ + typeOfData <- "/mcAll" + }else{ + typeOfData <- "/mcInd" + } + timeStepType <- paste0("/",paste(timeStep, classData, sep = "/")) + typeOfDataTime <- paste0(timeStepType, typeOfData) + nameStructure <- paste0(typeOfDataTime, "/structure") + + H5locAntaresh5 <- rhdf5::H5Fopen(name = pathH5) + resStruc <- rhdf5::h5ls(H5locAntaresh5) + dimData <- resStruc[ resStruc$group == typeOfDataTime & resStruc$name == "data", ]$dim + hourlyDataStructure <- rhdf5::h5read(H5locAntaresh5, name = nameStructure) + + indexCateroryInstance <- grep(instanceData, hourlyDataStructure[[classDataS]])[1] + + indexAntVar <- grep(antVar, hourlyDataStructure$variable)[1] + if(is.na(indexAntVar)){ + indexAntVar <- grep(antVar, hourlyDataStructure$reCalcVar)[1] + if(is.na(indexAntVar)){ + stop("error index") + }else{ + indexAntVar <- indexAntVar + length(hourlyDataStructure$variable) + } + } + if(allTimeId){ + maxTimeId <- as.integer(strsplit(dimData, "x")[[1]][1]) + indexTimeId <- 1:maxTimeId + }else{ + indexTimeId <- timeId + } + + if (is.null(mcYear)){ + indexMcYear <- 1 + }else{ + indexMcYear <- grep(mcYear, hourlyDataStructure$mcYear)[1] + } + + listIndex <- list(indexTimeId, indexAntVar, indexCateroryInstance, indexMcYear) + #debug print(listIndex) + + hourlyData <- rhdf5::h5read( + H5locAntaresh5, + name = paste0(typeOfDataTime, "/data"), + index = listIndex) + + hourlyData[,,,] <- newValue + + rhdf5::h5writeDataset.array( + obj = hourlyData, + h5loc = H5locAntaresh5, + name = paste0(typeOfDataTime, "/data"), + index = listIndex + ) + + rhdf5::H5Fclose(h5file = H5locAntaresh5) + rhdf5::h5closeAll() +} + + +#' Test opst +#' +#' @param test if x is simOptions class +#' +#' @noRd +.isSimOpts <- function(x){ + if ("simOptions" %in% class(x)){ + if (!is.null(x$h5path)){ + if (!file.exists(x$h5path)){ + warning(paste0("h5file does not exists for this study :", + x$studyName)) + return(FALSE) + }else{ + return(TRUE) + } + }else{ + #opts but no h5 (TXT) + return(TRUE) + } + }else{ + return(FALSE) + } +} diff --git a/appveyor.yml b/appveyor.yml index c1ef022..7e06aff 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -17,8 +17,10 @@ environment: matrix: - R_VERSION: release R_ARCH: x64 + USE_RTOOLS: true - R_VERSION: patched + USE_RTOOLS: true build_script: @@ -27,7 +29,7 @@ build_script: # - ./travis-tool.sh install_github Bioconductor-mirror/rhdf5 #after - travis-tool.sh install_bioc_deps - - ./travis-tool.sh install_github rte-antares-rpackage/antaresRead@next_release + - ./travis-tool.sh install_github rte-antares-rpackage/antaresRead@master test_script: - travis-tool.sh run_tests diff --git a/man/addConvergencePriceArea.Rd b/man/addConvergencePriceArea.Rd new file mode 100644 index 0000000..cb07c5b --- /dev/null +++ b/man/addConvergencePriceArea.Rd @@ -0,0 +1,33 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/addConvergencePriceArea.R +\name{addConvergencePriceArea} +\alias{addConvergencePriceArea} +\title{addConvergencePriceArea} +\usage{ +addConvergencePriceArea(antaresData = NULL) +} +\arguments{ +\item{antaresData}{Object of class \code{antaresData} created with function +\code{\link[antaresRead]{readAntares}}. antaresData must contains areas and +links details hourly data with linkCapacity.} +} +\description{ +This function computes priceConvergenceArea, priceConvergenceArea represent +the biggest system without congestion for one area. +} +\examples{ +\dontrun{ + + myData <- readAntares(areas = "all", + links = "all", + showProgress = FALSE, + linkCapacity = TRUE, + mcYears = "all") + + myDataRV <- removeVirtualAreas(x = myData, + storageFlexibility = getAreas(c("psp", "hub")), + production = getAreas("off")) + + addConvergencePriceArea(myData) +} +} diff --git a/man/addConvergencePriceSystem.Rd b/man/addConvergencePriceSystem.Rd new file mode 100644 index 0000000..315a964 --- /dev/null +++ b/man/addConvergencePriceSystem.Rd @@ -0,0 +1,33 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/addConvergencePriceSystem.R +\name{addConvergencePriceSystem} +\alias{addConvergencePriceSystem} +\title{addConvergencePriceSystem} +\usage{ +addConvergencePriceSystem(antaresData = NULL) +} +\arguments{ +\item{antaresData}{Object of class \code{antaresData} created with function +\code{\link[antaresRead]{readAntares}}. antaresData must contains areas and +links details data with linkCapacity.} +} +\description{ +This function computes priceConvergenceSystem, priceConvergenceSystem represent +the biggest system without congestion. +} +\examples{ +\dontrun{ + + myData <- readAntares(areas = "all", + links = "all", + showProgress = FALSE, + linkCapacity = TRUE, + mcYears = "all") + + myDataRV <- removeVirtualAreas(x = myData, + storageFlexibility = getAreas(c("psp", "hub")), + production = getAreas("off")) + + addConvergencePriceSystem(myData) +} +} diff --git a/man/addDownwardMargin.Rd b/man/addDownwardMargin.Rd index a062f09..2cddbbf 100644 --- a/man/addDownwardMargin.Rd +++ b/man/addDownwardMargin.Rd @@ -7,8 +7,7 @@ addDownwardMargin(x) } \arguments{ -\item{x}{An object of class \code{antaresData} created with -\code{\link[antaresRead]{readAntares}}} +\item{x}{An object of class \code{\link[antaresRead]{readAntares}} (or \code{\link[antaresRead]{simOptions}}) created with 'readAntares()' (or 'setSimulationPath()')} } \value{ The function modifies its input by adding to it two new columns diff --git a/man/addMonotones.Rd b/man/addMonotones.Rd new file mode 100644 index 0000000..b0a5923 --- /dev/null +++ b/man/addMonotones.Rd @@ -0,0 +1,34 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/addMonotones.R +\name{addMonotones} +\alias{addMonotones} +\title{addMonotones} +\usage{ +addMonotones(antaresData = NULL, variable = NULL) +} +\arguments{ +\item{antaresData}{Object of class \code{antaresData} created with function +\code{\link[antaresRead]{readAntares}}.} + +\item{variable}{An ANTARES variable.} +} +\value{ +\code{addMonotones} modifies its input by adding monotones. +} +\description{ +This function computes monotones for some variables. +} +\examples{ +\dontrun{ +# First simulation +studyPath <- "path/to/study/" + +setSimulationPath(studyPath, 1) +myData1 <- readAntares(areas = "all", +districts = "all", synthesis = FALSE) +addMonotones(antaresData = myData1, +variable = "LOAD") + +} + +} diff --git a/man/addProcessingH5.Rd b/man/addProcessingH5.Rd index fcd62fe..5239b19 100644 --- a/man/addProcessingH5.Rd +++ b/man/addProcessingH5.Rd @@ -7,9 +7,9 @@ addProcessingH5(opts = simOptions(), mcY = c("mcInd", "mcAll"), timeStep = "hourly", addNetLoad = FALSE, addDownwardMargin = FALSE, addUpwardMargin = FALSE, addExportAndImport = FALSE, - addLoadFactorLink = FALSE, externalDependency = FALSE, - loadFactor = FALSE, modulation = FALSE, netLoadRamp = FALSE, - surplus = FALSE, surplusClusters = FALSE, + addLoadFactorLink = FALSE, correctBalance = FALSE, + externalDependency = FALSE, loadFactor = FALSE, modulation = FALSE, + netLoadRamp = FALSE, surplus = FALSE, surplusClusters = FALSE, thermalAvailabilities = FALSE, linkCapacity = FALSE, mustRun = FALSE, allProcess = FALSE, evalAreas = list(), evalLinks = list(), evalClusters = list(), evalDistricts = list(), @@ -32,6 +32,8 @@ addProcessingH5(opts = simOptions(), mcY = c("mcInd", "mcAll"), \item{addLoadFactorLink}{\code{boolean} refer to \link[antaresProcessing]{addLoadFactorLink}} +\item{correctBalance}{\code{boolean} refer to \link[antaresProcessing]{correctBalance}} + \item{externalDependency}{\code{boolean} refer to \link[antaresProcessing]{externalDependency}} \item{loadFactor}{\code{boolean} refer to \link[antaresProcessing]{loadFactor}} @@ -92,6 +94,7 @@ addProcessingH5(opts = opts, mcY = "mcInd", addUpwardMargin = TRUE, addExportAndImport = TRUE, addLoadFactorLink = TRUE, + correctBalance = TRUE, externalDependency = TRUE, loadFactor = TRUE, modulation = TRUE, diff --git a/man/addUpwardMargin.Rd b/man/addUpwardMargin.Rd index 6c73a9b..4a3c173 100644 --- a/man/addUpwardMargin.Rd +++ b/man/addUpwardMargin.Rd @@ -7,8 +7,7 @@ addUpwardMargin(x) } \arguments{ -\item{x}{An object of class \code{antaresData} created with -\code{\link[antaresRead]{readAntares}}} +\item{x}{An object of class \code{\link[antaresRead]{readAntares}} (or \code{\link[antaresRead]{simOptions}}) created with 'readAntares()' (or 'setSimulationPath()')} } \value{ The function modifies its input by adding to it two new columns diff --git a/man/correctBalance.Rd b/man/correctBalance.Rd new file mode 100644 index 0000000..99fce9a --- /dev/null +++ b/man/correctBalance.Rd @@ -0,0 +1,34 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/correctBalance.R +\name{correctBalance} +\alias{correctBalance} +\title{correctBalance} +\usage{ +correctBalance(x) +} +\arguments{ +\item{x}{An object of class \code{\link[antaresRead]{readAntares}} (or \code{\link[antaresRead]{simOptions}}) created with 'readAntares()' (or 'setSimulationPath()')} +} +\value{ +\code{correctBalance} modifies its input by editing BALANCE and 'ROW BAL.'. +Formulas : +\enumerate{ +\item \eqn{BALANCE = BALANCE - 'ROW. BAL.'} +\item \eqn{ROW. BAL = 0} +} +} +\description{ +This function corrects the BALANCE with 'ROW BAL.'. +} +\examples{ +\dontrun{ +# First simulation +studyPath <- "path/to/study/" + +setSimulationPath(studyPath, 1) +mydata1 <- readAntares(areas = "all", districts = "all", synthesis = FALSE) +correctBalance(mydata1) + +} + +} diff --git a/man/neighbours.Rd b/man/neighbours.Rd new file mode 100644 index 0000000..bf10cf9 --- /dev/null +++ b/man/neighbours.Rd @@ -0,0 +1,47 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/neighbours.R +\name{neighbours} +\alias{neighbours} +\alias{addNeighbours} +\alias{getAllNeighbours} +\title{neighbours} +\usage{ +neighbours(areas = NULL, virtualAreas = NULL) + +addNeighbours(antaresData = NULL) + +getAllNeighbours(areasString = NULL, virtualAreas = NULL) +} +\arguments{ +\item{areas}{A vector with several areas names.} + +\item{virtualAreas}{A vector with several virtual areas names.} + +\item{antaresData}{Object of class \code{antaresData} created with function +\code{\link[antaresRead]{readAntares}}.} + +\item{areasString}{A string with several areas names separated by an espace, +see the examples.} +} +\value{ +\code{neighbours} return a vector with neighbours areas names. +\code{addNeighbours} modifies its input by adding a column neighbours. +\code{getAllNeighbours} return a vector with neighbours areas names. +} +\description{ +This function return a list of neighbours. +} +\examples{ +\dontrun{ + +res <- neighbours(areas = c("a", "c"), +virtualAreas = getAreas("psp")) + +myData <- readAntares(areas = c("a", "c"), links = getLinks("a"), +showProgress = FALSE) + +addNeighbours(myData) + +res <- getAllNeighbours(areasString = "a b") +} +} diff --git a/tests/testthat/helper_init.R b/tests/testthat/helper_init.R index 67df22c..f949c75 100644 --- a/tests/testthat/helper_init.R +++ b/tests/testthat/helper_init.R @@ -6,31 +6,6 @@ path <- tempdir() sourcedir <- system.file("inst/testdata", package = "antaresRead") if(sourcedir == ""){ sourcedir <- system.file("testdata", package = "antaresRead")} -check_if_h5_is_in_tmp<-function(h5filePath=NULL,path=NULL, stop=FALSE, printMessage=TRUE){ - - resH5NotInTmp<-!grepl("Temp", h5filePath, ignore.case = TRUE) & !grepl("tmp", h5filePath, ignore.case = TRUE) - if(resH5NotInTmp){ - if(printMessage){ - print(paste0("h5file : ", h5filePath)) - print(paste0("path : ", path)) - } - }else{ - return(TRUE) - } - - messageToPrint<-"h5file is not in temp folder" - if(stop & resH5NotInTmp){ - stop(messageToPrint) - } - if(resH5NotInTmp){ - if(printMessage){ - message(messageToPrint) - } - } - - return(FALSE) -} - Sys.unsetenv("R_TESTS") # Hack: For some unknown reason, this script is executed at some point of # the R CMD CHECK before package is correctly installed and tests actually run. @@ -57,11 +32,11 @@ if (sourcedir != "") { if(file.copy(from = h5file, to = path, overwrite = TRUE)){ assign("h5file", file.path(path, nameH5File), envir = globalenv()) #WE MUST assign h5file variable in the test environnement and not in the global environnement - if(!check_if_h5_is_in_tmp(h5file, path, printMessage = FALSE)){ + if(!.check_if_h5_is_in_tmp(h5file, path, printMessage = FALSE)){ assign("h5file", file.path(path, nameH5File)) } - check_if_h5_is_in_tmp(h5file, path) + .check_if_h5_is_in_tmp(h5file, path) } } @@ -75,7 +50,7 @@ if (sourcedir != "") { stop("h5file must not be null") } - check_if_h5_is_in_tmp(h5file, path, stop = FALSE) + .check_if_h5_is_in_tmp(h5file, path, stop = FALSE) silentf <- deprintize(showAliases) assign("silentf", silentf, envir = globalenv()) diff --git a/tests/testthat/test-addConvergencePriceArea.R b/tests/testthat/test-addConvergencePriceArea.R new file mode 100644 index 0000000..85e37cc --- /dev/null +++ b/tests/testthat/test-addConvergencePriceArea.R @@ -0,0 +1,93 @@ +context("Function addConvergencePriceArea") + +opts <- setSimulationPath(studyPath) + +test_that("addConvergencePriceArea works with an antaresDataList. + antaresData must contains areas and links details data with linkCapacity.", { + myData <- readAntares(clusters = "all", showProgress = FALSE) + expect_error(addConvergencePriceArea(myData)) + myData <- readAntares(areas = "all", showProgress = FALSE) + expect_error(addConvergencePriceArea(myData)) + myData <- readAntares(areas = "all", + links = "all", + showProgress = FALSE) + expect_error(addConvergencePriceArea(myData)) + myData <- readAntares(areas = "all", + links = "all", + showProgress = FALSE, + mcYears = "all") + expect_error(addConvergencePriceArea(myData)) + myData <- suppressWarnings(readAntares(areas = "all", + links = "all", + showProgress = FALSE, + linkCapacity = TRUE, + mcYears = "all")) + addConvergencePriceArea(myData) +}) + +test_that("addConvergencePriceArea add a column priceConvergenceArea + For each time, priceConvergenceArea represent the system without congestion + for one area", { + myData <- suppressWarnings(readAntares(areas = "all", + links = "all", + showProgress = FALSE, + linkCapacity = TRUE, + mcYears = "all")) + myDataRV <- removeVirtualAreas(x = myData, + storageFlexibility = getAreas(c("psp", "hub")), + production = getAreas("off")) + addConvergencePriceArea(myDataRV) + + expect_true("priceConvergenceArea" %in% names(myDataRV$areas)) + expect_true(is.character(class(myDataRV$areas$priceConvergenceArea))) + + #when there is priceConvergenceArea = "a b c" then we have the same prices + indexAll <- myDataRV$areas[priceConvergenceArea=="a b c", which = TRUE] + myDataPrice <- myDataRV$areas[indexAll, + .(area, mcYear, timeId, `MRG. PRICE`)] + IdCols <- getIdCols(myDataPrice) + #without mcYear + IdColsWA <- IdCols[IdCols!="area"] + myFormula <- sprintf("%s ~ area", paste(IdColsWA, collapse = "+")) + diffPrice <- data.table::dcast(data = myDataPrice, + as.formula(myFormula), + value.var = "MRG. PRICE") + diffPrice[, ':=' (diffBA = abs(b - a), + diffCB = abs(c - b)) ] + + maxHurdl <- max(myDataRV$links[link=="b - c", hurdlesCostDirect]) + maxHurdlInd <- max(myDataRV$links[link=="b - c", hurdlesCostIndirect]) + maxDiffNormal <- max(maxHurdl, maxHurdlInd) + #diff price not bigger than hurdle cost + expect_true(max(diffPrice$diffBA) <= maxDiffNormal) + expect_true(max(diffPrice$diffCB) <= maxDiffNormal) + + # when PriceAreaSystem == "a" then big difference between price + indexAB <- myDataRV$areas[priceConvergenceArea=="a b", which = TRUE] + myDataPrice <- myDataRV$areas[indexAB, + .(area, mcYear, timeId, `MRG. PRICE`)] + + diffPrice <- data.table::dcast(data = myDataPrice, + as.formula(myFormula), + value.var = "MRG. PRICE") + diffPrice[, ':=' (diffBA = abs(b - a))] + #diff price not bigger than hurdle cost for A / B but not for C / B + expect_true(max(diffPrice$diffBA) <= maxDiffNormal) + + #when priceConvergenceArea = "a - b" the area is a or b + expect_false("c" %in% unique(myDataRV$areas[priceConvergenceArea=="a b", + (area)])) + expect_true("a" %in% unique(myDataRV$areas[priceConvergenceArea=="a b", + (area)])) + expect_true("b" %in% unique(myDataRV$areas[priceConvergenceArea=="a b", + (area)])) + + #when priceConvergenceArea = "a" the area is a + expect_true("a" %in% unique(myDataRV$areas[priceConvergenceArea=="a", + (area)])) + expect_false("b" %in% unique(myDataRV$areas[priceConvergenceArea=="a", + (area)])) + expect_false("c" %in% unique(myDataRV$areas[priceConvergenceArea=="a", + (area)])) + +}) diff --git a/tests/testthat/test-addConvergencePriceSystem.R b/tests/testthat/test-addConvergencePriceSystem.R new file mode 100644 index 0000000..32dafc4 --- /dev/null +++ b/tests/testthat/test-addConvergencePriceSystem.R @@ -0,0 +1,80 @@ +context("Function addConvergencePriceSystem") + +opts <- setSimulationPath(studyPath) + +test_that("addConvergencePriceSystem works with an antaresDataList. + antaresData must contains areas and links details data with linkCapacity.", { + myData <- readAntares(clusters = "all", showProgress = FALSE) + expect_error(addConvergencePriceSystem(myData)) + myData <- readAntares(areas = "all", showProgress = FALSE) + expect_error(addConvergencePriceSystem(myData)) + myData <- readAntares(links = "all", showProgress = FALSE) + expect_error(addConvergencePriceSystem(myData)) + + myData <- readAntares(areas = "all", + links = "all", + showProgress = FALSE) + expect_error(addConvergencePriceSystem(myData)) + myData <- suppressWarnings(readAntares(areas = "all", + links = "all", + showProgress = FALSE, + linkCapacity = TRUE)) + expect_error(addConvergencePriceSystem(myData)) + myData <- suppressWarnings(readAntares(areas = "all", + links = "all", + showProgress = FALSE, + linkCapacity = TRUE, + mcYears = "all")) + addConvergencePriceSystem(myData) +}) + +test_that("addConvergencePriceSystem add a column priceConvergenceSystem. + For each time, priceConvergenceSystem represent the system without congestion", { + myData <- suppressWarnings(readAntares(areas = "all", + links = "all", + showProgress = FALSE, + linkCapacity = TRUE, + mcYears = "all")) + myDataRV <- removeVirtualAreas(x = myData, + storageFlexibility = getAreas(c("psp", "hub")), + production = getAreas("off")) + addConvergencePriceSystem(myDataRV) + + expect_true("priceConvergenceSystem" %in% names(myDataRV$areas)) + expect_true(is.character(class(myDataRV$areas$priceConvergenceSystem))) + + #when there is priceConvergenceSystem = "a b c" then we have the same prices + indexAll <- myDataRV$areas[priceConvergenceSystem=="a b c", which = TRUE] + myDataPrice <- myDataRV$areas[indexAll, + .(area, mcYear, timeId, `MRG. PRICE`)] + IdCols <- getIdCols(myDataPrice) + #without mcYear + IdColsWA <- IdCols[IdCols!="area"] + myFormula <- sprintf("%s ~ area", paste(IdColsWA, collapse = "+")) + diffPrice <- dcast(data = myDataPrice, + as.formula(myFormula), + value.var = "MRG. PRICE") + diffPrice[, ':=' (diffBA = abs(b - a), + diffCB = abs(c - b)) ] + + maxHurdl <- max(myDataRV$links[link=="b - c", hurdlesCostDirect]) + maxHurdlInd <- max(myDataRV$links[link=="b - c", hurdlesCostIndirect]) + maxDiffNormal <- max(maxHurdl, maxHurdlInd) + #diff price not bigger than hurdle cost + expect_true(max(diffPrice$diffBA) <= maxDiffNormal) + expect_true(max(diffPrice$diffCB) <= maxDiffNormal) + + # when PriceAreaSystem == "a" then big difference between price + indexAB <- myDataRV$areas[priceConvergenceSystem=="a b", which = TRUE] + myDataPrice <- myDataRV$areas[indexAB, + .(area, mcYear, timeId, `MRG. PRICE`)] + + diffPrice <- dcast(data = myDataPrice, + as.formula(myFormula), + value.var = "MRG. PRICE") + diffPrice[, ':=' (diffBA = abs(b - a), + diffCB = abs(c - b)) ] + #diff price not bigger than hurdle cost for A / B but not for C / B + expect_true(max(diffPrice$diffBA) <= maxDiffNormal) + expect_false(max(diffPrice$diffCB) <= maxDiffNormal) +}) diff --git a/tests/testthat/test-addMonotones.R b/tests/testthat/test-addMonotones.R new file mode 100644 index 0000000..9d7323d --- /dev/null +++ b/tests/testthat/test-addMonotones.R @@ -0,0 +1,136 @@ +context("Function addMonotones") + +opts <- setSimulationPath(studyPath) + +test_that("addMonotones works with an antaresData detail", { + expect_error(addMonotones(antaresData = 1e-10), + regexp = NULL) + + myDataS <- readAntares(showProgress = FALSE) + expect_error(addMonotones(antaresData = myDataS), + regexp = NULL) + +}) + +test_that("addMonotones works with a correct variable", { + myDataD <- readAntares(mcYears = "all", showProgress = FALSE) + expect_error(addMonotones(antaresData = myDataD, + variable = 3), + regexp = "variable is not a character") + myDataD <- readAntares(mcYears = "all", showProgress = FALSE) + expect_error(addMonotones(antaresData = myDataD, + variable = 3), + regexp = "variable is not a character") + myDataD <- readAntares(mcYears = "all", showProgress = FALSE, links = "all") + expect_error(addMonotones(antaresData = myDataD, + variable = "LOAD"), + regexp = "Incorrect variable") + myDataD <- readAntares(mcYears = "all", showProgress = FALSE, areas = "all") + addMonotones(antaresData = myDataD, + variable = "LOAD") +}) + +test_that("addMonotones works with an antaresDataTable", { + myDataD <- readAntares(mcYears = "all", showProgress = FALSE, areas = "all") + addMonotones(antaresData = myDataD, + variable = "LOAD") + expect_true("LOADMonoMc1" %in% names(myDataD)) + expect_true("LOADMonoMc2" %in% names(myDataD)) + expect_true("LOADMonoMean" %in% names(myDataD)) + expect_true(is.integer(myDataD$LOADMonoMc1)) + expect_true(is.integer(myDataD$LOADMonoMc2)) + expect_true(is.integer(myDataD$LOADMonoMean)) + myDataD <- readAntares(mcYears = "all", showProgress = FALSE, areas = "all") + addMonotones(antaresData = myDataD, + variable = "MRG. PRICE") + expect_true("MRG. PRICEMonoMc1" %in% names(myDataD)) + expect_true("MRG. PRICEMonoMc2" %in% names(myDataD)) + expect_true("MRG. PRICEMonoMean" %in% names(myDataD)) + expect_true(is.numeric(myDataD$`MRG. PRICEMonoMc1`)) + expect_true(is.numeric(myDataD$`MRG. PRICEMonoMc2`)) + expect_true(is.numeric(myDataD$`MRG. PRICEMonoMean`)) + + for(myArea in unique(myDataD$area)){ + priceAmc1 <- myDataD[area==myArea & mcYear==1, `MRG. PRICE` ] + priceAmc1Mono <- sort(priceAmc1, decreasing = TRUE) + expect_equal(myDataD[area==myArea & mcYear==1, `MRG. PRICEMonoMc1` ], + priceAmc1Mono) + } + for(myArea in unique(myDataD$area)){ + priceAmc1 <- myDataD[area==myArea & mcYear==2, `MRG. PRICE` ] + priceAmc1Mono <- sort(priceAmc1, decreasing = TRUE) + expect_equal(myDataD[area==myArea & mcYear==2, `MRG. PRICEMonoMc2` ], + priceAmc1Mono) + } + +}) + +test_that("addMonotones works with an antaresDataList", { + myDataD <- readAntares(mcYears = "all", + showProgress = FALSE, + areas = "all", + links = "all") + addMonotones(antaresData = myDataD, + variable = "LOAD") + expect_true("LOADMonoMc1" %in% names(myDataD$areas)) + expect_true("LOADMonoMc2" %in% names(myDataD$areas)) + expect_true("LOADMonoMean" %in% names(myDataD$areas)) + expect_true(is.integer(myDataD$areas$LOADMonoMc1)) + expect_true(is.integer(myDataD$areas$LOADMonoMc2)) + expect_true(is.integer(myDataD$areas$LOADMonoMean)) + myDataD <- readAntares(mcYears = "all", + showProgress = FALSE, + areas = "all", + links = "all") + addMonotones(antaresData = myDataD, + variable = "FLOW LIN.") + expect_true("FLOW LIN.MonoMc1" %in% names(myDataD$links)) + expect_true("FLOW LIN.MonoMc2" %in% names(myDataD$links)) + expect_true("FLOW LIN.MonoMean" %in% names(myDataD$links)) + expect_true(is.numeric(myDataD$links$`FLOW LIN.MonoMc1`)) + expect_true(is.numeric(myDataD$links$`FLOW LIN.MonoMc2`)) + expect_true(is.numeric(myDataD$links$`FLOW LIN.MonoMean`)) + + for(myLink in unique(myDataD$links$link)){ + flowLMc1 <- myDataD$links[link==myLink & mcYear==1, `FLOW LIN.` ] + flowLMc1Mono <- sort(flowLMc1, decreasing = TRUE) + expect_equal(myDataD$links[link==myLink & mcYear==1, `FLOW LIN.MonoMc1` ], + flowLMc1Mono) + } + for(myLink in unique(myDataD$links$link)){ + flowLMc2 <- myDataD$links[link==myLink & mcYear==2, `FLOW LIN.` ] + flowLMc2Mono <- sort(flowLMc2, decreasing = TRUE) + expect_equal(myDataD$links[link==myLink & mcYear==2, `FLOW LIN.MonoMc2` ], + flowLMc2Mono) + } + + myDataD <- readAntares(mcYears = "all", + showProgress = FALSE, + areas = "all", + districts = "all") + addMonotones(antaresData = myDataD, + variable = "LOAD") + expect_true("LOADMonoMc1" %in% names(myDataD$areas)) + expect_true("LOADMonoMc2" %in% names(myDataD$areas)) + expect_true("LOADMonoMean" %in% names(myDataD$areas)) + expect_true("LOADMonoMc1" %in% names(myDataD$districts)) + expect_true("LOADMonoMc2" %in% names(myDataD$districts)) + expect_true("LOADMonoMean" %in% names(myDataD$districts)) +}) + +test_that("addMonotones computes the same monotones for several years", { + myDataD <- readAntares(mcYears = "all", + showProgress = FALSE, + areas = "all") + addMonotones(antaresData = myDataD, + variable = "LOAD") + expect_true("LOADMonoMc1" %in% names(myDataD)) + expect_true("LOADMonoMc2" %in% names(myDataD)) + expect_true("LOADMonoMean" %in% names(myDataD)) + expect_equal(myDataD[mcYear==1, LOADMonoMc1], + myDataD[mcYear==2, LOADMonoMc1]) + expect_equal(myDataD[mcYear==1, LOADMonoMc2], + myDataD[mcYear==2, LOADMonoMc2]) + expect_equal(myDataD[mcYear==1, LOADMonoMean], + myDataD[mcYear==2, LOADMonoMean]) +}) diff --git a/tests/testthat/test-addUpwardMargin.R b/tests/testthat/test-addUpwardMargin.R index a901139..95cfe21 100644 --- a/tests/testthat/test-addUpwardMargin.R +++ b/tests/testthat/test-addUpwardMargin.R @@ -80,3 +80,20 @@ describe("addUpwardMargin", { }) }) + + +test_that("#32 addUpwardMargin must work with an antaresDataList without virtual areas", { + dataT <- suppressWarnings(readAntares( + areas = "all", links = "all", + select = "upwardMargin", + hydroStorageMaxPower = TRUE, linkCapacity = TRUE, + showProgress = FALSE + )) + + addUpwardMargin(dataT) + + expect_true("isolatedUpwardMargin" %in% names(dataT$areas)) + expect_true("interconnectedUpwardMargin" %in% names(dataT$areas)) + expect_gt(max(unique(dataT$areas$isolatedUpwardMargin)), 1) + expect_gt(max(unique(dataT$areas$interconnectedUpwardMargin)), 1) +}) diff --git a/tests/testthat/test-correctBalance.R b/tests/testthat/test-correctBalance.R new file mode 100644 index 0000000..243da71 --- /dev/null +++ b/tests/testthat/test-correctBalance.R @@ -0,0 +1,315 @@ +context("Function correctBalance") + +opts <- setSimulationPath(studyPath) + +test_that("correctBalance get an antaresData in x", { + expect_error(correctBalance(x = 1e-10)) +}) + +test_that("correctBalance must work with an antaresDataTable areas", { + myDataAreas <- readAntares(areas = "all", showProgress = FALSE, mcYears = 1) + myDataAreasCopy <- data.table::copy(myDataAreas) + expect_true(all.equal(unique(myDataAreasCopy$`ROW BAL.`), 0)) + vecCorect <- as.integer(round(rnorm(69, mean = 100, sd = 5),0)) + myDataAreasCopy[area == "b" & timeId > 2710 & timeId < 2780, `ROW BAL.` := vecCorect] + expect_false(identical(unique(myDataAreasCopy$`ROW BAL.`), 0)) + correctBalance(x = myDataAreasCopy) + expect_true(all.equal(unique(myDataAreasCopy$`ROW BAL.`), 0)) + expect_equal(unique(myDataAreasCopy$`ROW BAL.`), 0) + #check the Balance value + expect_equal(myDataAreasCopy[area == "b" & timeId > 2710 & timeId < 2780, (BALANCE)], + myDataAreas[area == "b" & timeId > 2710 & timeId < 2780, (BALANCE)] - vecCorect) + expect_true("oldBalance" %in% names(myDataAreasCopy)) +}) + +test_that("correctBalance must work with an antaresDataTable districts", { + myDataDistricts <- readAntares(districts = "all", showProgress = FALSE, mcYears = 1) + myDataDistrictCopy <- data.table::copy(myDataDistricts) + expect_true(all.equal(unique(myDataDistrictCopy$`ROW BAL.`), 0)) + vecCorect <- as.integer(round(rnorm(69, mean = 100, sd = 5),0)) + myDataDistrictCopy[district == getDistricts()[1] & timeId > 2710 & timeId < 2780, `ROW BAL.` := vecCorect] + expect_false(identical(unique(myDataDistrictCopy$`ROW BAL.`), 0)) + correctBalance(x = myDataDistrictCopy) + expect_true(all.equal(unique(myDataDistrictCopy$`ROW BAL.`), 0)) + expect_equal(unique(myDataDistrictCopy$`ROW BAL.`), 0) + #check the Balance value + expect_equal(myDataDistrictCopy[district == getDistricts()[1] & timeId > 2710 & timeId < 2780, (BALANCE)], + myDataDistricts[district == getDistricts()[1] & timeId > 2710 & timeId < 2780, (BALANCE)] - vecCorect) + + expect_true("oldBalance" %in% names(myDataDistrictCopy)) +}) + +test_that("correctBalance must work with an antaresDataList", { + myDataArDi <- readAntares(areas = "all", districts = "all", showProgress = FALSE, mcYears = 1) + myDataDistrictCopy <- data.table::copy(myDataArDi) + expect_true(all.equal(unique(myDataDistrictCopy$districts$`ROW BAL.`), 0)) + vecCorect <- as.integer(round(rnorm(69, mean = 100, sd = 5),0)) + myDataDistrictCopy$districts[district == getDistricts()[1] & timeId > 2710 & timeId < 2780, `ROW BAL.` := vecCorect] + expect_false(identical(unique(myDataDistrictCopy$districts$`ROW BAL.`), 0)) + corBalance <- correctBalance(x = myDataDistrictCopy) + expect_true(all.equal(unique(corBalance$districts$`ROW BAL.`), 0)) + expect_equal(unique(corBalance$districts$`ROW BAL.`), 0) + #check the Balance value + myDataDistrictCopy$districts[district == getDistricts()[1] & timeId > 2710 & timeId < 2780, (BALANCE)] + expect_equal(myDataDistrictCopy$districts[district == getDistricts()[1] & timeId > 2710 & timeId < 2780, (BALANCE)], + myDataArDi$district[district == getDistricts()[1] & timeId > 2710 & timeId < 2780, (BALANCE)] - vecCorect) + +}) + +test_that("correctBalance must work with an antaresDataTable links", { + myDataLinks <- readAntares(links = "all", showProgress = FALSE, mcYears = 1) + myDataLinksCopy <- data.table::copy(myDataLinks) + expect_true(all.equal(myDataLinksCopy, correctBalance(x = myDataLinksCopy))) +}) + +test_that("correctBalance must work with an antaresDataList without areas or districts", { + myDataLinksClusters <- readAntares(links = "all", clusters = "all", showProgress = FALSE, mcYears = 1) + myDataLinksClustersCopy <- data.table::copy(myDataLinksClusters) + expect_true(all.equal(myDataLinksClustersCopy, correctBalance(x = myDataLinksClustersCopy))) +}) + +test_that("h5 : processing correctBalance when mcYear is set to NULL", { + .skipFunctionH5(h5file) + optsH5 <- setSimulationPath(h5file) + correctBalance(x = optsH5) + # with a new Study H5 test + ## create a new folder h5 + pathInitial <- file.path(dirname(studyPath)) + pathNewH5 <- file.path(pathInitial, "testH5") + if (!dir.exists(pathNewH5)){ + dir.create(pathNewH5) + } + + #write a new H5 file + optsData <- antaresRead::setSimulationPath(path = studyPath) + suppressWarnings(writeAntaresH5(path = pathNewH5, opts = optsData, + overwrite = TRUE, supressMessages = TRUE)) + pathNewH5File <- file.path(pathNewH5, list.files(pathNewH5)) + .h5Antares_edit_variable( + pathH5 = pathNewH5File, + instanceData = "b", + classData = "areas", + timeId = 1:100, + antVar = "ROW BAL.", + newValue = 15000, + mcYear = NULL + ) + + optsH5New <- setSimulationPath(path = pathNewH5File) + myDataB <- readAntares(opts = optsH5New, areas = "b", mcYears = NULL) + expect_equal(max(unique(myDataB$`ROW BAL.`)), 15000) + correctBalance(x = optsH5New) + myDataB <- readAntares(opts = optsH5New, areas = "b", mcYears = NULL) + expect_equal(max(unique(myDataB$`ROW BAL.`)), 0) + +}) + +test_that("h5 : processing correctBalance when mcYear is set to 1", { + .skipFunctionH5(h5file) + optsH5 <- setSimulationPath(h5file) + correctBalance(x = optsH5) + # with a new Study H5 test + ## create a new folder h5 + pathInitial <- file.path(dirname(studyPath)) + pathNewH5 <- file.path(pathInitial, "testH5") + if (!dir.exists(pathNewH5)){ + dir.create(pathNewH5) + } + + #write a new H5 file + optsData <- antaresRead::setSimulationPath(path = studyPath) + suppressWarnings(writeAntaresH5(path = pathNewH5, opts = optsData, + overwrite = TRUE, supressMessages = TRUE)) + + pathNewH5File <- file.path(pathNewH5, list.files(pathNewH5)) + .h5Antares_edit_variable( + pathH5 = pathNewH5File, + instanceData = "b", + classData = "areas", + timeId = 1:100, + antVar = "ROW BAL.", + newValue = 15000, + mcYear = 1 + ) + + optsH5New <- setSimulationPath(path = pathNewH5File) + myDataB <- readAntares(opts = optsH5New, areas = "b", mcYears = 1) + expect_equal(max(unique(myDataB$`ROW BAL.`)), 15000) + correctBalance(x = optsH5New) + myDataB <- readAntares(opts = optsH5New, areas = "b", + mcYears = 1, select = c("BALANCE", "ROW BAL.", "oldBalance")) + expect_equal(max(unique(myDataB$`ROW BAL.`)), 0) + expect_true("oldBalance" %in% names(myDataB)) +}) + +test_that("h5 : processing correctBalance when mcYear is set to 1 for weekly or monthly data", { + .skipFunctionH5(h5file) + optsH5 <- setSimulationPath(h5file) + correctBalance(x = optsH5) + # with a new Study H5 test + ## create a new folder h5 + pathInitial <- file.path(dirname(studyPath)) + pathNewH5 <- file.path(pathInitial, "testH5") + if (!dir.exists(pathNewH5)){ + dir.create(pathNewH5) + } + + #write a new H5 file + optsData <- antaresRead::setSimulationPath(path = studyPath) + suppressWarnings(writeAntaresH5(path = pathNewH5, opts = optsData, + overwrite = TRUE, supressMessages = TRUE)) + + pathNewH5File <- file.path(pathNewH5, list.files(pathNewH5)) + .h5Antares_edit_variable( + pathH5 = pathNewH5File, + instanceData = "b", + classData = "areas", + timeId = 1, + antVar = "ROW BAL.", + newValue = 15000, + mcYear = 1, + timeStep = "weekly" + ) + + optsH5New <- setSimulationPath(path = pathNewH5File) + myDataB <- readAntares(opts = optsH5New, areas = "b", + mcYears = 1, timeStep = "weekly") + expect_equal(max(unique(myDataB$`ROW BAL.`)), 15000) + correctBalance(x = optsH5New) + myDataB <- readAntares(opts = optsH5New, areas = "b", + mcYears = 1, + select = c("BALANCE", "ROW BAL.", "oldBalance"), + timeStep = "weekly") + expect_equal(max(unique(myDataB$`ROW BAL.`)), 0) + expect_true("oldBalance" %in% names(myDataB)) + + # monthly data + timeStepToTest <- "monthly" + optsH5 <- antaresRead::setSimulationPath(path = h5file) + myDataB <- readAntares(opts = optsH5, + areas = "b", + mcYears = 1, timeStep = timeStepToTest) + expect_equal(max(unique(myDataB$`ROW BAL.`)), 0) + pathInitial <- file.path(dirname(studyPath)) + pathNewH5 <- file.path(pathInitial, "testH5") + if (!dir.exists(pathNewH5)){ + dir.create(pathNewH5) + } + #write a new H5 file + optsData <- antaresRead::setSimulationPath(path = studyPath) + suppressWarnings(writeAntaresH5(path = pathNewH5, opts = optsData, + overwrite = TRUE, supressMessages = TRUE)) + + pathNewH5File <- file.path(pathNewH5, list.files(pathNewH5)) + .h5Antares_edit_variable( + pathH5 = pathNewH5File, + instanceData = "b", + classData = "areas", + timeId = 1, + antVar = "ROW BAL.", + newValue = 15000, + mcYear = 1, + timeStep = timeStepToTest + ) + myDataB <- readAntares(opts = optsH5New, areas = "b", + mcYears = 1, timeStep = timeStepToTest) + expect_equal(max(unique(myDataB$`ROW BAL.`)), 15000) + correctBalance(x = optsH5New) + myDataB <- readAntares(opts = optsH5New, areas = "b", + mcYears = 1, + timeStep = timeStepToTest, + select = c("BALANCE", "ROW BAL.", "oldBalance")) + expect_equal(max(unique(myDataB$`ROW BAL.`)), 0) + expect_true("oldBalance" %in% names(myDataB)) +}) + +test_that("h5 : processing correctBalance when mcYear is set to 1 for monthly data on districts", { + .skipFunctionH5(h5file) + # monthly data + timeStepToTest <- "monthly" + optsH5 <- antaresRead::setSimulationPath(path = h5file) + districtToC <- getDistricts()[1] + myDataC <- readAntares(opts = optsH5, + districts = districtToC, + mcYears = 1, + timeStep = timeStepToTest) + expect_equal(max(unique(myDataC$`ROW BAL.`)), 0) + pathInitial <- file.path(dirname(studyPath)) + pathNewH5 <- file.path(pathInitial, "testH5") + if (!dir.exists(pathNewH5)){ + dir.create(pathNewH5) + } + #write a new H5 file + optsData <- antaresRead::setSimulationPath(path = studyPath) + suppressWarnings(writeAntaresH5(path = pathNewH5, opts = optsData, + overwrite = TRUE, supressMessages = TRUE)) + pathNewH5File <- file.path(pathNewH5, list.files(pathNewH5)) + + optsH5New <- setSimulationPath(path = pathNewH5File) + .h5Antares_edit_variable( + pathH5 = pathNewH5File, + instanceData = districtToC, + classData = "districts", + timeId = 1, + antVar = "ROW BAL.", + newValue = 15000, + mcYear = 1, + timeStep = timeStepToTest + ) + myDataC <- readAntares(opts = optsH5New, district = districtToC, + mcYears = 1, timeStep = timeStepToTest) + expect_equal(max(unique(myDataC$`ROW BAL.`)), 15000) + correctBalance(x = optsH5New) + myDataC <- readAntares(opts = optsH5New, + district = districtToC, + mcYears = 1, + timeStep = timeStepToTest, + select = c("BALANCE", "ROW BAL.", "oldBalance")) + expect_equal(max(unique(myDataC$`ROW BAL.`)), 0) + expect_true("oldBalance" %in% names(myDataC)) +}) + +test_that("h5 : processing correctBalance with addProcessingH5", { + .skipFunctionH5(h5file) + optsH5 <- setSimulationPath(h5file) + correctBalance(x = optsH5) + myData <- readAntares(opts = optsH5, areas = "b", mcYears = 1) + expect_equal(max(unique(myData$`ROW BAL.`)), 0) + # with a new Study H5 test + ## create a new folder h5 + pathInitial <- file.path(dirname(studyPath)) + pathNewH5 <- file.path(pathInitial, "testH5") + if (!dir.exists(pathNewH5)){ + dir.create(pathNewH5) + } + + #write a new H5 file + optsData <- antaresRead::setSimulationPath(path = studyPath) + suppressWarnings(writeAntaresH5(path = pathNewH5, opts = optsData, + overwrite = TRUE, supressMessages = TRUE)) + + pathNewH5File <- file.path(pathNewH5, list.files(pathNewH5)) + .h5Antares_edit_variable( + pathH5 = pathNewH5File, + instanceData = "b", + classData = "areas", + timeId = 1:100, + antVar = "ROW BAL.", + newValue = 15000, + mcYear = 1 + ) + + optsH5New <- setSimulationPath(path = pathNewH5File) + myDataB <- readAntares(opts = optsH5New, areas = "b", mcYears = 1) + expect_equal(max(unique(myDataB$`ROW BAL.`)), 15000) + + addProcessingH5(opts = optsH5New, + correctBalance = TRUE, + mcY = "mcInd") + + myDataB <- readAntares(opts = optsH5New, areas = "b", + mcYears = 1, select = c("BALANCE", "ROW BAL.", "oldBalance")) + expect_equal(max(unique(myDataB$`ROW BAL.`)), 0) + + expect_true("oldBalance" %in% names(myDataB)) +}) diff --git a/tests/testthat/test-h5_processing.R b/tests/testthat/test-h5_processing.R index 7b2ec36..f5935ca 100644 --- a/tests/testthat/test-h5_processing.R +++ b/tests/testthat/test-h5_processing.R @@ -1,22 +1,7 @@ context("h5 : addProcessingH5") -#write data -skipFunctionH5<-function(){ - skip_if(is.null(h5file), "h5file is null") - if(!is.null(h5file)){ - skip_if(!check_if_h5_is_in_tmp(h5file), "h5file is not in temp folder") - } - skip_if_not(.requireRhdf5_Antares(stopP = FALSE), - "please install a correct rhdf5 version") - - #I dont why but according to CRAN Team antaresProcessing try to write in the user library - # but there are checks to verify that h5 file is in tmp folder - #to comment in the futur - skip_on_cran() -} - test_that("h5 : processing, write results", { - skipFunctionH5() + .skipFunctionH5(h5file) .setAliasH5() optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd", @@ -25,6 +10,7 @@ test_that("h5 : processing, write results", { addUpwardMargin = TRUE, addExportAndImport = TRUE, addLoadFactorLink = TRUE, + correctBalance = TRUE, externalDependency = TRUE, loadFactor = TRUE, modulation = TRUE, @@ -41,7 +27,7 @@ test_that("h5 : processing, write results", { }) test_that("h5 : processing, write results mcAll", { - skipFunctionH5() + .skipFunctionH5(h5file) .setAliasH5() optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcAll", @@ -50,6 +36,7 @@ test_that("h5 : processing, write results mcAll", { addExportAndImport = TRUE, addLoadFactorLink = TRUE, externalDependency = TRUE, + correctBalance = TRUE, loadFactor = TRUE, modulation = TRUE, netLoadRamp = TRUE, @@ -66,7 +53,7 @@ test_that("h5 : processing, write results mcAll", { }) test_that("h5 : all data", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd", allProcess = TRUE)}) @@ -82,7 +69,7 @@ test_that("h5 : all data", { #read data and compare test_that("h5 : processing calc by user", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) calcData <- readAntares(areas = "all", mcYears = "all", select = c("H. STOR" , "MISC. DTG", @@ -96,7 +83,7 @@ test_that("h5 : processing calc by user", { }) test_that("h5 : processing calc by straitements", { - skipFunctionH5() + .skipFunctionH5(h5file) UpwardMargin_out <- readAntares(areas = "all", mcYears = "all", select = "Out_addUpwardMargin", showProgress = FALSE) @@ -112,7 +99,7 @@ test_that("h5 : processing calc by straitements", { }) test_that("h5 : processing calc links clusters discticcts", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd", evalLinks = list(totalLink= "`FLOW LIN.` + 10000 "), @@ -130,7 +117,7 @@ test_that("h5 : processing calc links clusters discticcts", { }) test_that("h5 : processing Out_addNetLoad", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",addNetLoad = TRUE)}) re <- readAntares(opts = optsH5, areas = "all", districts = "all", @@ -141,7 +128,7 @@ test_that("h5 : processing Out_addNetLoad", { }) test_that("h5 : processing Out_addDownwardMargin", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",addDownwardMargin = TRUE)}) re <- readAntares(opts = optsH5, areas = "all", districts = "all", @@ -154,7 +141,7 @@ test_that("h5 : processing Out_addDownwardMargin", { }) test_that("h5 : processing Out_addUpwardMargin", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",addUpwardMargin = TRUE)}) re <- readAntares(opts = optsH5, areas = "all", districts = "all", @@ -167,7 +154,7 @@ test_that("h5 : processing Out_addUpwardMargin", { }) test_that("h5 : processing Out_addExportAndImport", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",addExportAndImport = TRUE)}) re <- readAntares(opts = optsH5, areas = "all", districts = "all", @@ -180,7 +167,7 @@ test_that("h5 : processing Out_addExportAndImport", { }) test_that("h5 : processing Out_addLoadFactorLink", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",addLoadFactorLink = TRUE)}) re <- readAntares(opts = optsH5, links = "all",clusters = "all", @@ -190,7 +177,7 @@ test_that("h5 : processing Out_addLoadFactorLink", { }) test_that("h5 : processing Out_externalDependency", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",externalDependency = TRUE)}) re <- readAntares(opts = optsH5, areas = "all", districts = "all", @@ -210,7 +197,7 @@ test_that("h5 : processing Out_externalDependency", { }) test_that("h5 : processing Out_loadFactor", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",loadFactor = TRUE)}) re <- readAntares(opts = optsH5, clusters = "all", @@ -222,7 +209,7 @@ test_that("h5 : processing Out_loadFactor", { }) test_that("h5 : processing Out_modulation", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd", modulation = TRUE)}) @@ -235,7 +222,7 @@ test_that("h5 : processing Out_modulation", { }) test_that("h5 : processing netLoadRamp", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",netLoadRamp = TRUE)}) @@ -252,7 +239,7 @@ test_that("h5 : processing netLoadRamp", { }) test_that("h5 : processing surplus", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",surplus = TRUE)}) @@ -275,7 +262,7 @@ test_that("h5 : processing surplus", { }) test_that("h5 : Out_surplusClusters", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings({addProcessingH5(opts = optsH5, mcY = "mcInd",surplusClusters = TRUE)}) @@ -291,7 +278,7 @@ test_that("h5 : Out_surplusClusters", { }) test_that("No h5 opts", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) optsH5$h5path <- NULL optsH5$h5 <- NULL @@ -300,14 +287,14 @@ test_that("No h5 opts", { }) test_that("Write boolean", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) suppressWarnings(addProcessingH5(optsH5, evalAreas = list(toto = "TRUE"))) expect_false(is.null(readAntares(mcYears = 1, select = "toto", showProgress = FALSE)$toto)) }) test_that("Write character", { - skipFunctionH5() + .skipFunctionH5(h5file) optsH5 <- setSimulationPath(h5file) expect_error(addProcessingH5(optsH5, evalAreas = list(toto = "'GGG'"))) }) diff --git a/tests/testthat/test-neighbours.R b/tests/testthat/test-neighbours.R new file mode 100644 index 0000000..7352b47 --- /dev/null +++ b/tests/testthat/test-neighbours.R @@ -0,0 +1,37 @@ +context("Function neighbours") + +opts <- setSimulationPath(studyPath) + +test_that("neighbours get neighbours of an area", { + res <- antaresProcessing::neighbours(areas = c("a", "c"), + virtualAreas = getAreas("psp")) + expect_true(grepl("a_offshore", x = res[area=="a"]$neighbours)) + expect_true(grepl("b", x = res[area=="a"]$neighbours)) + expect_true(grepl("b", x = res[area=="c"]$neighbours)) +}) + + +test_that("addNeighbours add neighbours to antaresData", { + myData <- antaresRead::readAntares(links = getLinks("c"), showProgress = FALSE) + expect_error(addNeighbours(myData)) + myData <- antaresRead::readAntares(areas = c("a", "c"), showProgress = FALSE) + addNeighbours(myData) + expect_true("neighbours" %in% names(addNeighbours(myData))) + neiC <- unique(myData[area=="c", (neighbours)]) + expect_true(grepl("b", neiC)) + expect_true(grepl("hub", neiC)) + myData <- antaresRead::readAntares(areas = c("a", "c"), + links = getLinks("a"), + showProgress = FALSE) + addNeighbours(myData) + neiA <- unique(myData$areas[area=="a", (neighbours)]) + expect_true(grepl("b", neiA)) + expect_true(grepl("a_offshore", neiA)) +}) + +test_that("getAllNeighbours get all neighbours", { + res <- getAllNeighbours(areasString = "a b") + expect_true("a_offshore" %in% res) + expect_true("c" %in% res) +}) +