Skip to content

Commit

Permalink
Merge branch 'develop' into unit-testing-2
Browse files Browse the repository at this point in the history
  • Loading branch information
meetagrawal09 authored Jul 27, 2024
2 parents 8ccdc6d + f72abcc commit 6cc0392
Show file tree
Hide file tree
Showing 21 changed files with 500 additions and 687 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ For more information about this file see also [Keep a Changelog](http://keepacha
- Added new feature of preparing initial conditions for MODIS LAI, AGB, ISCN SOC, and soil moisture across NA anchor sites.
- Added GEDI AGB preparation workflow.
- Added new feature of downloading datasets from the NASA DAAC ORNL database.
- Added 'aggregate' as a new feature for downscaled data

### Fixed

Expand Down
4 changes: 3 additions & 1 deletion CITATION.cff
Original file line number Diff line number Diff line change
Expand Up @@ -120,7 +120,9 @@ authors:
- given-names: Eric R. Scott
affiliation: University of Arizona
orcid: 'https://orcid.org/0000-0002-7430-7879'

- given-names: Harunobu Ishii
affiliation: Boston University Software & Application Innovation Lab(SAIL)

preferred-citation:
type: article
title: Facilitating feedbacks between field measurements and ecosystem models
Expand Down
2 changes: 0 additions & 2 deletions docker-compose.dev.yml
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,6 @@
#
# docker-compose -f docker-compose.yml -f docker-compose.dev.yml

version: '3.2'

services:

# web application. This expects the config.php to be copied from docker/web
Expand Down
1 change: 0 additions & 1 deletion docker-compose.https.yml
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
# Use this file to enable https
version: "3.2"

services:
traefik:
Expand Down
15 changes: 13 additions & 2 deletions docker-compose.yml
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
version: "3.2"

services:

# webserver to handle all traffic. This can use let's encrypt to generate a SSL cert.
Expand Down Expand Up @@ -206,6 +204,19 @@ services:
# ----------------------------------------------------------------------
# PEcAn models, list each model you want to run below
# ----------------------------------------------------------------------
# PEcAn FATES model runner
fates:
user: "${UID:-1001}:${GID:-1001}"
image: ghcr.io/noresmhub/ctsm-api:latest
restart: unless-stopped
networks:
- pecan
environment:
- RABBITMQ_URI=${RABBITMQ_URI:-amqp://guest:guest@rabbitmq/%2F}
depends_on:
- rabbitmq
volumes:
- pecan:/data

# PEcAn basgra model runner
basgra:
Expand Down
2 changes: 2 additions & 0 deletions docker/depends/pecan_package_dependencies.csv
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,7 @@
"dplyr",">= 1.1.2","base/db","Imports",FALSE
"ellipse","*","modules/assim.batch","Imports",FALSE
"emdbook","*","modules/assim.sequential","Suggests",FALSE
"exactextractr","*","modules/assim.sequential","Suggests",FALSE
"foreach","*","base/remote","Imports",FALSE
"foreach","*","modules/data.atmosphere","Suggests",FALSE
"foreach","*","modules/data.remote","Imports",FALSE
Expand Down Expand Up @@ -609,6 +610,7 @@
"testthat",">= 3.0.4","base/qaqc","Suggests",FALSE
"tibble","*","base/db","Imports",FALSE
"tibble","*","models/ed","Imports",FALSE
"tibble","*","models/fates","Imports",FALSE
"tibble","*","models/lpjguess","Imports",FALSE
"tibble","*","modules/data.atmosphere","Imports",FALSE
"tibble","*","modules/data.remote","Suggests",FALSE
Expand Down
4 changes: 0 additions & 4 deletions docker/docker-compose.example.yml
Original file line number Diff line number Diff line change
@@ -1,7 +1,3 @@
# you need a version entry, and this should be the same version as the
# docker-compose.yml file.
version: "3"

# if you change any of the services you will need the services header.
services:

Expand Down
3 changes: 2 additions & 1 deletion models/fates/DESCRIPTION
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,8 @@ Imports:
PEcAn.remote,
PEcAn.utils,
lubridate (>= 1.6.0),
ncdf4 (>= 1.15)
ncdf4 (>= 1.15),
tibble
Suggests:
testthat (>= 1.0.2)
License: BSD_3_clause + file LICENSE
Expand Down
253 changes: 139 additions & 114 deletions models/fates/R/met2model.FATES.R
100644 → 100755
Original file line number Diff line number Diff line change
Expand Up @@ -16,148 +16,173 @@
##' @param in.path location on disk where inputs are stored
##' @param in.prefix prefix of input and output files
##' @param outfolder location on disk where outputs will be stored
##' @param start_date the start date of the data to be downloaded (will only use the year part of the date)
##' @param end_date the end date of the data to be downloaded (will only use the year part of the date)
##' @param start_date the start date of the data to be downloaded
##' @param end_date the end date of the data to be downloaded
##' @param lst timezone offset to GMT in hours
##' @param overwrite should existing files be overwritten
##' @param verbose should the function be very verbosefor(year in start_year:end_year)
##' @param verbose should the function be very verbose for(year in start_year:end_year)
##' @importFrom ncdf4 ncvar_get ncdim_def ncatt_get ncvar_put
met2model.FATES <- function(in.path, in.prefix, outfolder, start_date, end_date, lst = 0, lat, lon,
overwrite = FALSE, verbose = FALSE, ...) {


met2model.FATES <- function(in.path,in.prefix,outfolder,start_date,end_date,lst=0,lat, lon, overwrite = FALSE, verbose = FALSE, ...) {
# General Structure- FATES Uses Netcdf so we need to rename vars, split files from years into months, and generate the header file
# Get Met file from inpath.
# Loop over years (Open nc.file,rename vars,change dimensions as needed,close/save .nc file)
# close
# defining temporal dimension needs to be figured out. If we configure FATES to use same tstep then we may not need to change dimensions


insert <- function(ncout, name, unit, data) {
var <- ncdf4::ncvar_def(name = name, units = unit, dim = dim, missval = -6999, verbose = verbose)
ncout <- ncdf4::ncvar_add(nc = ncout, v = var, verbose = verbose)
ncvar_put(nc = ncout, varid = name, vals = data)
insert <- function(ncout, name, unit, data, dim) {
var <- ncdf4::ncvar_def(name, unit, dim = dim, missval = as.numeric(1.0e36), verbose = verbose)
ncout <- ncdf4::ncvar_add(ncout, var)
ncdf4::ncvar_put(nc = ncout, varid = name, vals = data)
return(invisible(ncout))
}
sm <- c(0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365) * 86400 ## day of year thresholds

## Create output directory
dir.create(outfolder)
if (!file.exists(outfolder)){
dir.create(outfolder)
}

# Process start and end dates
start_date <- as.POSIXlt(start_date, tz = "UTC")
end_date <- as.POSIXlt(end_date, tz = "UTC")
## Process start, end dates
start_date <- as.POSIXlt(start_date, tz = "UTC", origin = "1700-01-01")
end_date <- as.POSIXlt(end_date, tz = "UTC", origin = "1700-01-01")
start_year <- lubridate::year(start_date)
end_year <- lubridate::year(end_date)

## Build met
for (year in start_year:end_year) {


## Process time
base_time <- difftime(paste0(year,"-01-01"),"1700-01-01", units="days") ## days of the year
if (lubridate::leap_year(year)){ # True
sm <- c(0, 31, 58, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365)
}
else {
sm <- c(0, 31, 59, 91, 121, 152, 182, 213, 244, 274, 305, 335, 366)
}

in.file <- file.path(in.path, paste(in.prefix, year, "nc", sep = "."))

if (file.exists(in.file)) {

## Open netcdf file
nc <- ncdf4::nc_open(in.file)

## extract variables. These need to be read in and converted to CLM names (all units are correct)
time <- ncvar_get(nc, "time")
latitude <- ncvar_get(nc, "latitude")
longitude <- ncvar_get(nc, "longitude")
FLDS <- ncvar_get(nc, "surface_downwelling_longwave_flux_in_air") ## W/m2
FSDS <- ncvar_get(nc, "surface_downwelling_shortwave_flux_in_air") ## W/m2
PRECTmms <- ncvar_get(nc, "precipitation_flux") ## kg/m2/s -> mm/s (same val, diff name)
PSRF <- ncvar_get(nc, "air_pressure") ## Pa
SHUM <- ncvar_get(nc, "specific_humidity") ## g/g -> kg/kg
TBOT <- ncvar_get(nc, "air_temperature") ## K
WIND <- sqrt(ncvar_get(nc, "eastward_wind") ^ 2 + ncvar_get(nc, "northward_wind") ^ 2) ## m/s
time <- ncdf4::ncvar_get(nc, "time")
LATIXY <- ncdf4::ncvar_get(nc, "latitude")
LONGXY <- ncdf4::ncvar_get(nc, "longitude")
FLDS <- ncdf4::ncvar_get(nc, "surface_downwelling_longwave_flux_in_air") ## W/m2
FSDS <- ncdf4::ncvar_get(nc, "surface_downwelling_shortwave_flux_in_air") ## W/m2
PRECTmms <- ncdf4::ncvar_get(nc, "precipitation_flux") ## kg/m2/s -> mm/s (same val, diff name)
PSRF <- ncdf4::ncvar_get(nc, "air_pressure") ## Pa
QBOT <- ncdf4::ncvar_get(nc, "specific_humidity") ## g/g -> kg/kg
TBOT <- ncdf4::ncvar_get(nc, "air_temperature") ## K
WIND <- sqrt(ncdf4::ncvar_get(nc, "eastward_wind") ^ 2 + ncdf4::ncvar_get(nc, "northward_wind") ^ 2) ## m/s

## CREATE MONTHLY FILES
for (mo in 1:12) {
tsel <- which(time > sm[mo] & time <= sm[mo + 1])
outfile <- file.path(outfolder, paste0(formatC(year, width = 4, flag = "0"), "-",
formatC(mo, width = 2, flag = "0"), ".nc"))
if (file.exists(outfile) & overwrite == FALSE) {
if (((year==start_year) & (mo<lubridate::month(start_date))) | ((year==end_year) & (mo>lubridate::month(end_date)))){
next
}

lat.dim <- ncdim_def(name = "latitude", units = "", vals = 1:1, create_dimvar = FALSE)
lon.dim <- ncdim_def(name = "longitude", units = "", vals = 1:1, create_dimvar = FALSE)
time.dim <- ncdim_def(name = "time", units = "seconds", vals = time,
create_dimvar = TRUE, unlim = TRUE)
dim <- list(lat.dim, lon.dim, time.dim) ## docs say this should be time,lat,lon but get error writing unlimited first
## http://www.cesm.ucar.edu/models/cesm1.2/clm/models/lnd/clm/doc/UsersGuide/x12979.html

# LATITUDE
var <- ncdf4::ncvar_def(name = "latitude", units = "degree_north",
dim = list(lat.dim, lon.dim), missval = as.numeric(-9999))
ncout <- ncdf4::nc_create(outfile, vars = var, verbose = verbose)
ncvar_put(nc = ncout, varid = "latitude", vals = latitude)

# LONGITUDE
var <- ncdf4::ncvar_def(name = "longitude", units = "degree_east",
dim = list(lat.dim, lon.dim), missval = as.numeric(-9999))
ncout <- ncdf4::ncvar_add(nc = ncout, v = var, verbose = verbose)
ncvar_put(nc = ncout, varid = "longitude", vals = longitude)

## surface_downwelling_longwave_flux_in_air
ncout <- insert(ncout, "FLDS", "W m-2", FLDS)

## surface_downwelling_shortwave_flux_in_air
ncout <- insert(ncout, "FSDS", "W m-2", FSDS)

## precipitation_flux
ncout <- insert(ncout, "PRECTmms", "mm/s", PRECTmms)

## air_pressure
ncout <- insert(ncout, "PSRF", "Pa", PSRF)

## specific_humidity
ncout <- insert(ncout, "SHUM", "kg/kg", SHUM)

## air_temperature
ncout <- insert(ncout, "TBOT", "K", TBOT)

## eastward_wind & northward_wind
ncout <- insert(ncout, "WIND", "m/s", WIND)

ncdf4::nc_close(ncout)

# ncvar_rename(ncfile,varid="LONGXY")
# ncvar_rename(ncfile,varid="LATIXY")
# #
# # double EDGEW(scalar) ;
# # EDGEW:long_name = "western edge in atmospheric data" ;
# # EDGEW:units = "degrees E" ;
# EDGEW = ncvar_rename(ncfile,"EDGEW","EDGEW")
#
# # double EDGEE(scalar) ;
# # EDGEE:long_name = "eastern edge in atmospheric data" ;
# # EDGEE:units = "degrees E" ;
# EDGEE = ncvar_rename(ncfile,"EDGEE","EDGEE")
#
# # double EDGES(scalar) ;
# # EDGES:long_name = "southern edge in atmospheric data" ;
# # EDGES:units = "degrees N" ;
# EDGES = ncvar_rename(ncfile,"EDGES","EDGES")
# #
# # double EDGEN(scalar) ;
# # EDGEN:long_name = "northern edge in atmospheric data" ;
# # EDGEN:units = "degrees N" ;
# EDGEN = ncvar_rename(ncfile,"EDGEN","EDGEN")
else {
# slice
tsel <- which(time > base_time+sm[mo] & time <= base_time+sm[mo+1])
print(mo)
if (length(tsel)!=0){
# define dim
lat.dim <- ncdf4::ncdim_def(name = "lat", units = "", vals = 1:1, create_dimvar=FALSE)
lon.dim <- ncdf4::ncdim_def(name = "lon", units = "", vals = 1:1, create_dimvar=FALSE)
time.dim <- ncdf4::ncdim_def(name = "time", units = "", vals = 1:length(time[tsel]),create_dimvar = TRUE, calendar="standard", unlim = FALSE) #left to CTSM automatically transfer
scalar.dim <- ncdf4::ncdim_def(name="scalar", units = "", vals = 1:1)
dim <- list(time.dim, lat.dim, lon.dim)

# LATITUDE
var_lat <- ncdf4::ncvar_def(name = "LATIXY", units = "degree_north",
dim = list(lat.dim, lon.dim), missval = as.numeric(-9999))
# LONGITUDE
var_long <- ncdf4::ncvar_def(name = "LONGXY", units = "degree_east",
dim = list(lat.dim, lon.dim), missval = as.numeric(-9999))
# time
var_time <- ncdf4::ncvar_def(name = "time", units = "days since 1700-01-01", prec = "float",
dim = list(time.dim), missval = as.numeric(-9999))
# EDGEE
var_E <- ncdf4::ncvar_def(name = "EDGEE", units = "degrees_east",
dim = list(scalar.dim, lat.dim, lon.dim), missval = as.numeric(-9999))
# EDGEW edge for resolution , edge-central 0.005, # PEcAn provide range of grid?
var_W <- ncdf4::ncvar_def(name = "EDGEW", units = "degrees_west",
dim = list(scalar.dim, lat.dim, lon.dim), missval = as.numeric(-9999))
# EDGES
var_S <- ncdf4::ncvar_def(name = "EDGES", units = "degrees_south",
dim = list(scalar.dim, lat.dim, lon.dim), missval = as.numeric(-9999))
# EDGEN
var_N <- ncdf4::ncvar_def(name = "EDGEN", units = "degrees_north",
dim = list(scalar.dim, lat.dim, lon.dim), missval = as.numeric(-9999))

## SAPERATELY CREATE FILES
put_var <- function(ncout){
ncdf4::ncvar_put(nc = ncout, varid = "LATIXY", vals = LATIXY) #same with FATES
ncdf4::ncvar_put(nc = ncout, varid = "LONGXY", vals = LONGXY)
ncdf4::ncvar_put(nc = ncout, varid = "EDGEE", vals = LONGXY+0.005)
ncdf4::ncvar_put(nc = ncout, varid = "EDGEW", vals = LONGXY-0.005)
ncdf4::ncvar_put(nc = ncout, varid = "EDGES", vals = LATIXY-0.005)
ncdf4::ncvar_put(nc = ncout, varid = "EDGEN", vals = LATIXY+0.005)
}
## Precipitation
outfile_prec <- file.path(outfolder, paste0("Prec", formatC(year, width = 4, flag = "0"), "-",
formatC(mo, width = 2, flag = "0"), ".nc"))
if (file.exists(outfile_prec) & overwrite == FALSE) {
next
}
ncout_prec <- ncdf4::nc_create(outfile_prec, vars = list(var_lat,var_long,var_E,var_W,var_S,var_N), verbose = verbose)
put_var(ncout_prec)
## precipitation_flux
ncout_prec <- insert(ncout_prec, "PRECTmms", "mm/s", PRECTmms[tsel], dim)
ncdf4::nc_close(ncout_prec)

## Solar
outfile_slr <- file.path(outfolder, paste0("Slr", formatC(year, width = 4, flag = "0"), "-",
formatC(mo, width = 2, flag = "0"), ".nc"))
if (file.exists(outfile_slr) & overwrite == FALSE) {
next
}
ncout_slr <- ncdf4::nc_create(outfile_slr, vars = list(var_lat,var_long,var_E,var_W,var_S,var_N), verbose = verbose)
put_var(ncout_slr)
## surface_downwelling_shortwave_flux_in_air
ncout_slr <- insert(ncout_slr, "FSDS", "W m-2", FSDS[tsel], dim)
ncdf4::nc_close(ncout_slr)

## Temerature and humidity
outfile_tem <- file.path(outfolder, paste0("Tem", formatC(year, width = 4, flag = "0"), "-",
formatC(mo, width = 2, flag = "0"), ".nc"))
if (file.exists(outfile_tem) & overwrite == FALSE) {
next
}
ncout_tem <- ncdf4::nc_create(outfile_tem, vars = list(var_lat,var_long,var_E,var_W,var_S,var_N), verbose = verbose)
put_var(ncout_tem)
## surface_downwelling_longwave_flux_in_air
ncout_tem <- insert(ncout_tem, "FLDS", "W m-2", FLDS[tsel], dim)
## air_pressure
ncout_tem <- insert(ncout_tem, "PSRF", "Pa", PSRF[tsel], dim)
## specific_humidity
ncout_tem <- insert(ncout_tem, "QBOT", "kg/kg", QBOT[tsel], dim)
## air_temperature
ncout_tem <- insert(ncout_tem, "TBOT", "K", TBOT[tsel], dim)
## eastward_wind & northward_wind
ncout_tem <- insert(ncout_tem, "WIND", "m/s", WIND[tsel], dim)
ncdf4::nc_close(ncout_tem)
}
}
}

ncdf4::nc_close(nc)
} ## end file exists
} ### end loop over met files
ncdf4::nc_close(nc)
} ## end input file
} ## end year loop over met files
results <- data.frame(file = paste0(outfolder, "/"),
host = c(PEcAn.remote::fqdn()),
mimetype = c("application/x-netcdf"),
formatname = c("CLM met"),
startdate = c(start_date),
enddate = c(end_date),
dbfile.name = "",
stringsAsFactors = FALSE)

PEcAn.logger::logger.info("Done with met2model.FATES")

return(data.frame(file = paste0(outfolder, "/"),
host = c(PEcAn.remote::fqdn()),
mimetype = c("application/x-netcdf"),
formatname = c("CLM met"),
startdate = c(start_date),
enddate = c(end_date),
dbfile.name = "",
stringsAsFactors = FALSE))
return(invisible(results))
} # met2model.FATES
Loading

0 comments on commit 6cc0392

Please sign in to comment.